1 /////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/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, Santiago Palacios 
   8 // Copyright:   (c) Michael Bedward (mbedward@ozemail.com.au) 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // For compilers that support precompilation, includes "wx/wx.h". 
  13 #include "wx/wxprec.h" 
  25     #include "wx/dcclient.h" 
  26     #include "wx/settings.h" 
  28     #include "wx/textctrl.h" 
  29     #include "wx/checkbox.h" 
  30     #include "wx/combobox.h" 
  31     #include "wx/valtext.h" 
  34     #include "wx/listbox.h" 
  37 #include "wx/textfile.h" 
  38 #include "wx/spinctrl.h" 
  39 #include "wx/tokenzr.h" 
  40 #include "wx/renderer.h" 
  42 #include "wx/generic/gridsel.h" 
  44 const wxChar wxGridNameStr
[] = wxT("grid"); 
  46 #if defined(__WXMOTIF__) 
  47     #define WXUNUSED_MOTIF(identifier)  WXUNUSED(identifier) 
  49     #define WXUNUSED_MOTIF(identifier)  identifier 
  52 #if defined(__WXGTK__) 
  53     #define WXUNUSED_GTK(identifier)    WXUNUSED(identifier) 
  55     #define WXUNUSED_GTK(identifier)    identifier 
  58 // Required for wxIs... functions 
  61 // ---------------------------------------------------------------------------- 
  63 // ---------------------------------------------------------------------------- 
  65 WX_DEFINE_ARRAY_WITH_DECL_PTR(wxGridCellAttr 
*, wxArrayAttrs
, 
  66                                  class WXDLLIMPEXP_ADV
); 
  68 struct wxGridCellWithAttr
 
  70     wxGridCellWithAttr(int row
, int col
, wxGridCellAttr 
*attr_
) 
  71         : coords(row
, col
), attr(attr_
) 
  80     wxGridCellCoords coords
; 
  84 //  DECLARE_NO_COPY_CLASS(wxGridCellWithAttr) 
  85 // without rewriting the macros, which require a public copy constructor. 
  88 WX_DECLARE_OBJARRAY_WITH_DECL(wxGridCellWithAttr
, wxGridCellWithAttrArray
, 
  89                               class WXDLLIMPEXP_ADV
); 
  91 #include "wx/arrimpl.cpp" 
  93 WX_DEFINE_OBJARRAY(wxGridCellCoordsArray
) 
  94 WX_DEFINE_OBJARRAY(wxGridCellWithAttrArray
) 
  96 // ---------------------------------------------------------------------------- 
  98 // ---------------------------------------------------------------------------- 
 100 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_CLICK
) 
 101 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_CLICK
) 
 102 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_DCLICK
) 
 103 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_DCLICK
) 
 104 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_BEGIN_DRAG
) 
 105 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_CLICK
) 
 106 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_CLICK
) 
 107 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_DCLICK
) 
 108 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_DCLICK
) 
 109 DEFINE_EVENT_TYPE(wxEVT_GRID_ROW_SIZE
) 
 110 DEFINE_EVENT_TYPE(wxEVT_GRID_COL_SIZE
) 
 111 DEFINE_EVENT_TYPE(wxEVT_GRID_COL_MOVE
) 
 112 DEFINE_EVENT_TYPE(wxEVT_GRID_RANGE_SELECT
) 
 113 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_CHANGE
) 
 114 DEFINE_EVENT_TYPE(wxEVT_GRID_SELECT_CELL
) 
 115 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_SHOWN
) 
 116 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_HIDDEN
) 
 117 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_CREATED
) 
 119 // ---------------------------------------------------------------------------- 
 121 // ---------------------------------------------------------------------------- 
 123 class WXDLLIMPEXP_ADV wxGridRowLabelWindow 
: public wxWindow
 
 126     wxGridRowLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 127     wxGridRowLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 128                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 133     void OnPaint( wxPaintEvent
& event 
); 
 134     void OnMouseEvent( wxMouseEvent
& event 
); 
 135     void OnMouseWheel( wxMouseEvent
& event 
); 
 136     void OnKeyDown( wxKeyEvent
& event 
); 
 137     void OnKeyUp( wxKeyEvent
& ); 
 138     void OnChar( wxKeyEvent
& ); 
 140     DECLARE_DYNAMIC_CLASS(wxGridRowLabelWindow
) 
 141     DECLARE_EVENT_TABLE() 
 142     DECLARE_NO_COPY_CLASS(wxGridRowLabelWindow
) 
 146 class WXDLLIMPEXP_ADV wxGridColLabelWindow 
: public wxWindow
 
 149     wxGridColLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 150     wxGridColLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 151                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 156     void OnPaint( wxPaintEvent
& event 
); 
 157     void OnMouseEvent( wxMouseEvent
& event 
); 
 158     void OnMouseWheel( wxMouseEvent
& event 
); 
 159     void OnKeyDown( wxKeyEvent
& event 
); 
 160     void OnKeyUp( wxKeyEvent
& ); 
 161     void OnChar( wxKeyEvent
& ); 
 163     DECLARE_DYNAMIC_CLASS(wxGridColLabelWindow
) 
 164     DECLARE_EVENT_TABLE() 
 165     DECLARE_NO_COPY_CLASS(wxGridColLabelWindow
) 
 169 class WXDLLIMPEXP_ADV wxGridCornerLabelWindow 
: public wxWindow
 
 172     wxGridCornerLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 173     wxGridCornerLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 174                              const wxPoint 
&pos
, const wxSize 
&size 
); 
 179     void OnMouseEvent( wxMouseEvent
& event 
); 
 180     void OnMouseWheel( wxMouseEvent
& event 
); 
 181     void OnKeyDown( wxKeyEvent
& event 
); 
 182     void OnKeyUp( wxKeyEvent
& ); 
 183     void OnChar( wxKeyEvent
& ); 
 184     void OnPaint( wxPaintEvent
& event 
); 
 186     DECLARE_DYNAMIC_CLASS(wxGridCornerLabelWindow
) 
 187     DECLARE_EVENT_TABLE() 
 188     DECLARE_NO_COPY_CLASS(wxGridCornerLabelWindow
) 
 191 class WXDLLIMPEXP_ADV wxGridWindow 
: public wxWindow
 
 197         m_rowLabelWin 
= NULL
; 
 198         m_colLabelWin 
= NULL
; 
 201     wxGridWindow( wxGrid 
*parent
, 
 202                   wxGridRowLabelWindow 
*rowLblWin
, 
 203                   wxGridColLabelWindow 
*colLblWin
, 
 204                   wxWindowID id
, const wxPoint 
&pos
, const wxSize 
&size 
); 
 205     virtual ~wxGridWindow() {} 
 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 OnChar( wxKeyEvent
& ); 
 222     void OnEraseBackground( wxEraseEvent
& ); 
 223     void OnFocus( wxFocusEvent
& ); 
 225     DECLARE_DYNAMIC_CLASS(wxGridWindow
) 
 226     DECLARE_EVENT_TABLE() 
 227     DECLARE_NO_COPY_CLASS(wxGridWindow
) 
 231 class wxGridCellEditorEvtHandler 
: public wxEvtHandler
 
 234     wxGridCellEditorEvtHandler(wxGrid
* grid
, wxGridCellEditor
* editor
) 
 241     void OnKillFocus(wxFocusEvent
& event
); 
 242     void OnKeyDown(wxKeyEvent
& event
); 
 243     void OnChar(wxKeyEvent
& event
); 
 245     void SetInSetFocus(bool inSetFocus
) { m_inSetFocus 
= inSetFocus
; } 
 249     wxGridCellEditor   
*m_editor
; 
 251     // Work around the fact that a focus kill event can be sent to 
 252     // a combobox within a set focus event. 
 255     DECLARE_EVENT_TABLE() 
 256     DECLARE_DYNAMIC_CLASS(wxGridCellEditorEvtHandler
) 
 257     DECLARE_NO_COPY_CLASS(wxGridCellEditorEvtHandler
) 
 261 IMPLEMENT_ABSTRACT_CLASS(wxGridCellEditorEvtHandler
, wxEvtHandler
) 
 263 BEGIN_EVENT_TABLE( wxGridCellEditorEvtHandler
, wxEvtHandler 
) 
 264     EVT_KILL_FOCUS( wxGridCellEditorEvtHandler::OnKillFocus 
) 
 265     EVT_KEY_DOWN( wxGridCellEditorEvtHandler::OnKeyDown 
) 
 266     EVT_CHAR( wxGridCellEditorEvtHandler::OnChar 
) 
 270 // ---------------------------------------------------------------------------- 
 271 // the internal data representation used by wxGridCellAttrProvider 
 272 // ---------------------------------------------------------------------------- 
 274 // this class stores attributes set for cells 
 275 class WXDLLIMPEXP_ADV wxGridCellAttrData
 
 278     void SetAttr(wxGridCellAttr 
*attr
, int row
, int col
); 
 279     wxGridCellAttr 
*GetAttr(int row
, int col
) const; 
 280     void UpdateAttrRows( size_t pos
, int numRows 
); 
 281     void UpdateAttrCols( size_t pos
, int numCols 
); 
 284     // searches for the attr for given cell, returns wxNOT_FOUND if not found 
 285     int FindIndex(int row
, int col
) const; 
 287     wxGridCellWithAttrArray m_attrs
; 
 290 // this class stores attributes set for rows or columns 
 291 class WXDLLIMPEXP_ADV wxGridRowOrColAttrData
 
 294     // empty ctor to suppress warnings 
 295     wxGridRowOrColAttrData() {} 
 296     ~wxGridRowOrColAttrData(); 
 298     void SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
); 
 299     wxGridCellAttr 
*GetAttr(int rowOrCol
) const; 
 300     void UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
); 
 303     wxArrayInt m_rowsOrCols
; 
 304     wxArrayAttrs m_attrs
; 
 307 // NB: this is just a wrapper around 3 objects: one which stores cell 
 308 //     attributes, and 2 others for row/col ones 
 309 class WXDLLIMPEXP_ADV wxGridCellAttrProviderData
 
 312     wxGridCellAttrData m_cellAttrs
; 
 313     wxGridRowOrColAttrData m_rowAttrs
, 
 318 // ---------------------------------------------------------------------------- 
 319 // data structures used for the data type registry 
 320 // ---------------------------------------------------------------------------- 
 322 struct wxGridDataTypeInfo
 
 324     wxGridDataTypeInfo(const wxString
& typeName
, 
 325                        wxGridCellRenderer
* renderer
, 
 326                        wxGridCellEditor
* editor
) 
 327         : m_typeName(typeName
), m_renderer(renderer
), m_editor(editor
) 
 330     ~wxGridDataTypeInfo() 
 332         wxSafeDecRef(m_renderer
); 
 333         wxSafeDecRef(m_editor
); 
 337     wxGridCellRenderer
* m_renderer
; 
 338     wxGridCellEditor
*   m_editor
; 
 340     DECLARE_NO_COPY_CLASS(wxGridDataTypeInfo
) 
 344 WX_DEFINE_ARRAY_WITH_DECL_PTR(wxGridDataTypeInfo
*, wxGridDataTypeInfoArray
, 
 345                                  class WXDLLIMPEXP_ADV
); 
 348 class WXDLLIMPEXP_ADV wxGridTypeRegistry
 
 351   wxGridTypeRegistry() {} 
 352     ~wxGridTypeRegistry(); 
 354     void RegisterDataType(const wxString
& typeName
, 
 355                      wxGridCellRenderer
* renderer
, 
 356                      wxGridCellEditor
* editor
); 
 358     // find one of already registered data types 
 359     int FindRegisteredDataType(const wxString
& typeName
); 
 361     // try to FindRegisteredDataType(), if this fails and typeName is one of 
 362     // standard typenames, register it and return its index 
 363     int FindDataType(const wxString
& typeName
); 
 365     // try to FindDataType(), if it fails see if it is not one of already 
 366     // registered data types with some params in which case clone the 
 367     // registered data type and set params for it 
 368     int FindOrCloneDataType(const wxString
& typeName
); 
 370     wxGridCellRenderer
* GetRenderer(int index
); 
 371     wxGridCellEditor
*   GetEditor(int index
); 
 374     wxGridDataTypeInfoArray m_typeinfo
; 
 378 // ---------------------------------------------------------------------------- 
 379 // conditional compilation 
 380 // ---------------------------------------------------------------------------- 
 382 #ifndef WXGRID_DRAW_LINES 
 383 #define WXGRID_DRAW_LINES 1 
 386 // ---------------------------------------------------------------------------- 
 388 // ---------------------------------------------------------------------------- 
 390 //#define DEBUG_ATTR_CACHE 
 391 #ifdef DEBUG_ATTR_CACHE 
 392     static size_t gs_nAttrCacheHits 
= 0; 
 393     static size_t gs_nAttrCacheMisses 
= 0; 
 396 // ---------------------------------------------------------------------------- 
 398 // ---------------------------------------------------------------------------- 
 400 wxGridCellCoords 
wxGridNoCellCoords( -1, -1 ); 
 401 wxRect 
wxGridNoCellRect( -1, -1, -1, -1 ); 
 404 // TODO: this doesn't work at all, grid cells have different sizes and approx 
 405 //       calculations don't work as because of the size mismatch scrollbars 
 406 //       sometimes fail to be shown when they should be or vice versa 
 408 //       The scroll bars may be a little flakey once in a while, but that is 
 409 //       surely much less horrible than having scroll lines of only 1!!! 
 412 //       Well, it's still seriously broken so it might be better but needs 
 415 static const size_t GRID_SCROLL_LINE_X 
= 15;  // 1; 
 416 static const size_t GRID_SCROLL_LINE_Y 
= GRID_SCROLL_LINE_X
; 
 418 // the size of hash tables used a bit everywhere (the max number of elements 
 419 // in these hash tables is the number of rows/columns) 
 420 static const int GRID_HASH_SIZE 
= 100; 
 423 // ---------------------------------------------------------------------------- 
 425 // ---------------------------------------------------------------------------- 
 427 static inline int GetScrollX(int x
) 
 429     return (x 
+ GRID_SCROLL_LINE_X 
- 1) / GRID_SCROLL_LINE_X
; 
 432 static inline int GetScrollY(int y
) 
 434     return (y 
+ GRID_SCROLL_LINE_Y 
- 1) / GRID_SCROLL_LINE_Y
; 
 438 // ============================================================================ 
 440 // ============================================================================ 
 442 // ---------------------------------------------------------------------------- 
 444 // ---------------------------------------------------------------------------- 
 446 wxGridCellEditor::wxGridCellEditor() 
 452 wxGridCellEditor::~wxGridCellEditor() 
 457 void wxGridCellEditor::Create(wxWindow
* WXUNUSED(parent
), 
 458                               wxWindowID 
WXUNUSED(id
), 
 459                               wxEvtHandler
* evtHandler
) 
 462         m_control
->PushEventHandler(evtHandler
); 
 465 void wxGridCellEditor::PaintBackground(const wxRect
& rectCell
, 
 466                                        wxGridCellAttr 
*attr
) 
 468     // erase the background because we might not fill the cell 
 469     wxClientDC 
dc(m_control
->GetParent()); 
 470     wxGridWindow
* gridWindow 
= wxDynamicCast(m_control
->GetParent(), wxGridWindow
); 
 472         gridWindow
->GetOwner()->PrepareDC(dc
); 
 474     dc
.SetPen(*wxTRANSPARENT_PEN
); 
 475     dc
.SetBrush(wxBrush(attr
->GetBackgroundColour(), wxSOLID
)); 
 476     dc
.DrawRectangle(rectCell
); 
 478     // redraw the control we just painted over 
 479     m_control
->Refresh(); 
 482 void wxGridCellEditor::Destroy() 
 486         m_control
->PopEventHandler( true /* delete it*/ ); 
 488         m_control
->Destroy(); 
 493 void wxGridCellEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
 495     wxASSERT_MSG(m_control
, wxT("The wxGridCellEditor must be created first!")); 
 497     m_control
->Show(show
); 
 501         // set the colours/fonts if we have any 
 504             m_colFgOld 
= m_control
->GetForegroundColour(); 
 505             m_control
->SetForegroundColour(attr
->GetTextColour()); 
 507             m_colBgOld 
= m_control
->GetBackgroundColour(); 
 508             m_control
->SetBackgroundColour(attr
->GetBackgroundColour()); 
 510 // Workaround for GTK+1 font setting problem on some platforms 
 511 #if !defined(__WXGTK__) || defined(__WXGTK20__) 
 512             m_fontOld 
= m_control
->GetFont(); 
 513             m_control
->SetFont(attr
->GetFont()); 
 516             // can't do anything more in the base class version, the other 
 517             // attributes may only be used by the derived classes 
 522         // restore the standard colours fonts 
 523         if ( m_colFgOld
.Ok() ) 
 525             m_control
->SetForegroundColour(m_colFgOld
); 
 526             m_colFgOld 
= wxNullColour
; 
 529         if ( m_colBgOld
.Ok() ) 
 531             m_control
->SetBackgroundColour(m_colBgOld
); 
 532             m_colBgOld 
= wxNullColour
; 
 535 // Workaround for GTK+1 font setting problem on some platforms 
 536 #if !defined(__WXGTK__) || defined(__WXGTK20__) 
 537         if ( m_fontOld
.Ok() ) 
 539             m_control
->SetFont(m_fontOld
); 
 540             m_fontOld 
= wxNullFont
; 
 546 void wxGridCellEditor::SetSize(const wxRect
& rect
) 
 548     wxASSERT_MSG(m_control
, wxT("The wxGridCellEditor must be created first!")); 
 550     m_control
->SetSize(rect
, wxSIZE_ALLOW_MINUS_ONE
); 
 553 void wxGridCellEditor::HandleReturn(wxKeyEvent
& event
) 
 558 bool wxGridCellEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 560     bool ctrl 
= event
.ControlDown(); 
 561     bool alt  
= event
.AltDown(); 
 564     // On the Mac the Alt key is more like shift and is used for entry of 
 565     // valid characters, so check for Ctrl and Meta instead. 
 566     alt 
= event
.MetaDown(); 
 569     // Assume it's not a valid char if ctrl or alt is down, but if both are 
 570     // down then it may be because of an AltGr key combination, so let them 
 571     // through in that case. 
 572     if ((ctrl 
|| alt
) && !(ctrl 
&& alt
)) 
 579     // If it's a F-Key or other special key then it shouldn't start the 
 581     if (event
.GetKeyCode() >= WXK_START
) 
 585     // if the unicode key code is not really a unicode character (it may 
 586     // be a function key or etc., the platforms appear to always give us a 
 587     // small value in this case) then fallback to the ASCII key code but 
 588     // don't do anything for function keys or etc. 
 589     key 
= event
.GetUnicodeKey(); 
 592         key 
= event
.GetKeyCode(); 
 593         keyOk 
= (key 
<= 127); 
 596     key 
= event
.GetKeyCode(); 
 597     keyOk 
= (key 
<= 255); 
 603 void wxGridCellEditor::StartingKey(wxKeyEvent
& event
) 
 608 void wxGridCellEditor::StartingClick() 
 614 // ---------------------------------------------------------------------------- 
 615 // wxGridCellTextEditor 
 616 // ---------------------------------------------------------------------------- 
 618 wxGridCellTextEditor::wxGridCellTextEditor() 
 623 void wxGridCellTextEditor::Create(wxWindow
* parent
, 
 625                                   wxEvtHandler
* evtHandler
) 
 627     m_control 
= new wxTextCtrl(parent
, id
, wxEmptyString
, 
 628                                wxDefaultPosition
, wxDefaultSize
 
 629 #if defined(__WXMSW__) 
 630                                , wxTE_PROCESS_TAB 
| wxTE_AUTO_SCROLL 
| wxNO_BORDER
 
 634     // set max length allowed in the textctrl, if the parameter was set 
 637         ((wxTextCtrl
*)m_control
)->SetMaxLength(m_maxChars
); 
 640     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 643 void wxGridCellTextEditor::PaintBackground(const wxRect
& WXUNUSED(rectCell
), 
 644                                            wxGridCellAttr 
* WXUNUSED(attr
)) 
 646     // as we fill the entire client area, 
 647     // don't do anything here to minimize flicker 
 650 void wxGridCellTextEditor::SetSize(const wxRect
& rectOrig
) 
 652     wxRect 
rect(rectOrig
); 
 654     // Make the edit control large enough to allow for internal margins 
 656     // TODO: remove this if the text ctrl sizing is improved esp. for unix 
 658 #if defined(__WXGTK__) 
 666 #elif defined(__WXMSW__) 
 680     int extra_x 
= ( rect
.x 
> 2 ) ? 2 : 1; 
 681     int extra_y 
= ( rect
.y 
> 2 ) ? 2 : 1; 
 683     #if defined(__WXMOTIF__) 
 688     rect
.SetLeft( wxMax(0, rect
.x 
- extra_x
) ); 
 689     rect
.SetTop( wxMax(0, rect
.y 
- extra_y
) ); 
 690     rect
.SetRight( rect
.GetRight() + 2 * extra_x 
); 
 691     rect
.SetBottom( rect
.GetBottom() + 2 * extra_y 
); 
 694     wxGridCellEditor::SetSize(rect
); 
 697 void wxGridCellTextEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 699     wxASSERT_MSG(m_control
, wxT("The wxGridCellEditor must be created first!")); 
 701     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
 703     DoBeginEdit(m_startValue
); 
 706 void wxGridCellTextEditor::DoBeginEdit(const wxString
& startValue
) 
 708     Text()->SetValue(startValue
); 
 709     Text()->SetInsertionPointEnd(); 
 710     Text()->SetSelection(-1, -1); 
 714 bool wxGridCellTextEditor::EndEdit(int row
, int col
, wxGrid
* grid
) 
 716     wxASSERT_MSG(m_control
, wxT("The wxGridCellEditor must be created first!")); 
 718     bool changed 
= false; 
 719     wxString value 
= Text()->GetValue(); 
 720     if (value 
!= m_startValue
) 
 724         grid
->GetTable()->SetValue(row
, col
, value
); 
 726     m_startValue 
= wxEmptyString
; 
 728     // No point in setting the text of the hidden control 
 729     //Text()->SetValue(m_startValue); 
 734 void wxGridCellTextEditor::Reset() 
 736     wxASSERT_MSG(m_control
, wxT("The wxGridCellEditor must be created first!")); 
 738     DoReset(m_startValue
); 
 741 void wxGridCellTextEditor::DoReset(const wxString
& startValue
) 
 743     Text()->SetValue(startValue
); 
 744     Text()->SetInsertionPointEnd(); 
 747 bool wxGridCellTextEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 749     return wxGridCellEditor::IsAcceptedKey(event
); 
 752 void wxGridCellTextEditor::StartingKey(wxKeyEvent
& event
) 
 754     // Since this is now happening in the EVT_CHAR event EmulateKeyPress is no 
 755     // longer an appropriate way to get the character into the text control. 
 756     // Do it ourselves instead.  We know that if we get this far that we have 
 757     // a valid character, so not a whole lot of testing needs to be done. 
 759     wxTextCtrl
* tc 
= Text(); 
 764     ch 
= event
.GetUnicodeKey(); 
 766         ch 
= (wxChar
)event
.GetKeyCode(); 
 768     ch 
= (wxChar
)event
.GetKeyCode(); 
 774             // delete the character at the cursor 
 775             pos 
= tc
->GetInsertionPoint(); 
 776             if (pos 
< tc
->GetLastPosition()) 
 777                 tc
->Remove(pos
, pos 
+ 1); 
 781             // delete the character before the cursor 
 782             pos 
= tc
->GetInsertionPoint(); 
 784                 tc
->Remove(pos 
- 1, pos
); 
 793 void wxGridCellTextEditor::HandleReturn( wxKeyEvent
& 
 794                                          WXUNUSED_GTK(WXUNUSED_MOTIF(event
)) ) 
 796 #if defined(__WXMOTIF__) || defined(__WXGTK__) 
 797     // wxMotif needs a little extra help... 
 798     size_t pos 
= (size_t)( Text()->GetInsertionPoint() ); 
 799     wxString 
s( Text()->GetValue() ); 
 800     s 
= s
.Left(pos
) + wxT("\n") + s
.Mid(pos
); 
 802     Text()->SetInsertionPoint( pos 
); 
 804     // the other ports can handle a Return key press 
 810 void wxGridCellTextEditor::SetParameters(const wxString
& params
) 
 820         if ( params
.ToLong(&tmp
) ) 
 822             m_maxChars 
= (size_t)tmp
; 
 826             wxLogDebug( _T("Invalid wxGridCellTextEditor parameter string '%s' ignored"), params
.c_str() ); 
 831 // return the value in the text control 
 832 wxString 
wxGridCellTextEditor::GetValue() const 
 834     return Text()->GetValue(); 
 837 // ---------------------------------------------------------------------------- 
 838 // wxGridCellNumberEditor 
 839 // ---------------------------------------------------------------------------- 
 841 wxGridCellNumberEditor::wxGridCellNumberEditor(int min
, int max
) 
 847 void wxGridCellNumberEditor::Create(wxWindow
* parent
, 
 849                                     wxEvtHandler
* evtHandler
) 
 854         // create a spin ctrl 
 855         m_control 
= new wxSpinCtrl(parent
, wxID_ANY
, wxEmptyString
, 
 856                                    wxDefaultPosition
, wxDefaultSize
, 
 860         wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 865         // just a text control 
 866         wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
 869         Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
 874 void wxGridCellNumberEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 876     // first get the value 
 877     wxGridTableBase 
*table 
= grid
->GetTable(); 
 878     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
 880         m_valueOld 
= table
->GetValueAsLong(row
, col
); 
 885         wxString sValue 
= table
->GetValue(row
, col
); 
 886         if (! sValue
.ToLong(&m_valueOld
) && ! sValue
.empty()) 
 888             wxFAIL_MSG( _T("this cell doesn't have numeric value") ); 
 896         Spin()->SetValue((int)m_valueOld
); 
 902         DoBeginEdit(GetString()); 
 906 bool wxGridCellNumberEditor::EndEdit(int row
, int col
, 
 916         value 
= Spin()->GetValue(); 
 917         changed 
= value 
!= m_valueOld
; 
 919             text 
= wxString::Format(wxT("%ld"), value
); 
 924         text 
= Text()->GetValue(); 
 925         changed 
= (text
.empty() || text
.ToLong(&value
)) && (value 
!= m_valueOld
); 
 930         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_NUMBER
)) 
 931             grid
->GetTable()->SetValueAsLong(row
, col
, value
); 
 933             grid
->GetTable()->SetValue(row
, col
, text
); 
 939 void wxGridCellNumberEditor::Reset() 
 944         Spin()->SetValue((int)m_valueOld
); 
 949         DoReset(GetString()); 
 953 bool wxGridCellNumberEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 955     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
 957         int keycode 
= event
.GetKeyCode(); 
 958         if ( (keycode 
< 128) && 
 959              (wxIsdigit(keycode
) || keycode 
== '+' || keycode 
== '-')) 
 968 void wxGridCellNumberEditor::StartingKey(wxKeyEvent
& event
) 
 970     int keycode 
= event
.GetKeyCode(); 
 973         if ( wxIsdigit(keycode
) || keycode 
== '+' || keycode 
== '-') 
 975             wxGridCellTextEditor::StartingKey(event
); 
 984         if ( wxIsdigit(keycode
) ) 
 986             wxSpinCtrl
* spin 
= (wxSpinCtrl
*)m_control
; 
 987             spin
->SetValue(keycode 
- '0'); 
 988             spin
->SetSelection(1,1); 
 997 void wxGridCellNumberEditor::SetParameters(const wxString
& params
) 
1008         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
1012             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
1016                 // skip the error message below 
1021         wxLogDebug(_T("Invalid wxGridCellNumberEditor parameter string '%s' ignored"), params
.c_str()); 
1025 // return the value in the spin control if it is there (the text control otherwise) 
1026 wxString 
wxGridCellNumberEditor::GetValue() const 
1033         long value 
= Spin()->GetValue(); 
1034         s
.Printf(wxT("%ld"), value
); 
1039         s 
= Text()->GetValue(); 
1045 // ---------------------------------------------------------------------------- 
1046 // wxGridCellFloatEditor 
1047 // ---------------------------------------------------------------------------- 
1049 wxGridCellFloatEditor::wxGridCellFloatEditor(int width
, int precision
) 
1052     m_precision 
= precision
; 
1055 void wxGridCellFloatEditor::Create(wxWindow
* parent
, 
1057                                    wxEvtHandler
* evtHandler
) 
1059     wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
1061 #if wxUSE_VALIDATORS 
1062     Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
1066 void wxGridCellFloatEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1068     // first get the value 
1069     wxGridTableBase 
*table 
= grid
->GetTable(); 
1070     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
1072         m_valueOld 
= table
->GetValueAsDouble(row
, col
); 
1077         wxString sValue 
= table
->GetValue(row
, col
); 
1078         if (! sValue
.ToDouble(&m_valueOld
) && ! sValue
.empty()) 
1080             wxFAIL_MSG( _T("this cell doesn't have float value") ); 
1085     DoBeginEdit(GetString()); 
1088 bool wxGridCellFloatEditor::EndEdit(int row
, int col
, 
1092     wxString 
text(Text()->GetValue()); 
1094     if ( (text
.empty() || text
.ToDouble(&value
)) && 
1095             !wxIsSameDouble(value
, m_valueOld
) ) 
1097         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_FLOAT
)) 
1098             grid
->GetTable()->SetValueAsDouble(row
, col
, value
); 
1100             grid
->GetTable()->SetValue(row
, col
, text
); 
1108 void wxGridCellFloatEditor::Reset() 
1110     DoReset(GetString()); 
1113 void wxGridCellFloatEditor::StartingKey(wxKeyEvent
& event
) 
1115     int keycode 
= event
.GetKeyCode(); 
1117     tmpbuf
[0] = (char) keycode
; 
1119     wxString 
strbuf(tmpbuf
, *wxConvCurrent
); 
1122     bool is_decimal_point 
= ( strbuf 
== 
1123        wxLocale::GetInfo(wxLOCALE_DECIMAL_POINT
, wxLOCALE_CAT_NUMBER
) ); 
1125     bool is_decimal_point 
= ( strbuf 
== _T(".") ); 
1128     if ( wxIsdigit(keycode
) || keycode 
== '+' || keycode 
== '-' 
1129          || is_decimal_point 
) 
1131         wxGridCellTextEditor::StartingKey(event
); 
1133         // skip Skip() below 
1140 void wxGridCellFloatEditor::SetParameters(const wxString
& params
) 
1151         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
1155             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
1157                 m_precision 
= (int)tmp
; 
1159                 // skip the error message below 
1164         wxLogDebug(_T("Invalid wxGridCellFloatEditor parameter string '%s' ignored"), params
.c_str()); 
1168 wxString 
wxGridCellFloatEditor::GetString() const 
1171     if ( m_precision 
== -1 && m_width 
!= -1) 
1173         // default precision 
1174         fmt
.Printf(_T("%%%d.f"), m_width
); 
1176     else if ( m_precision 
!= -1 && m_width 
== -1) 
1179         fmt
.Printf(_T("%%.%df"), m_precision
); 
1181     else if ( m_precision 
!= -1 && m_width 
!= -1 ) 
1183         fmt
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
1187         // default width/precision 
1191     return wxString::Format(fmt
, m_valueOld
); 
1194 bool wxGridCellFloatEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1196     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1198         const int keycode 
= event
.GetKeyCode(); 
1199         if ( isascii(keycode
) ) 
1202             tmpbuf
[0] = (char) keycode
; 
1204             wxString 
strbuf(tmpbuf
, *wxConvCurrent
); 
1207             const wxString decimalPoint 
= 
1208                 wxLocale::GetInfo(wxLOCALE_DECIMAL_POINT
, wxLOCALE_CAT_NUMBER
); 
1210             const wxString 
decimalPoint(_T('.')); 
1213             // accept digits, 'e' as in '1e+6', also '-', '+', and '.' 
1214             if ( wxIsdigit(keycode
) || 
1215                     tolower(keycode
) == 'e' || 
1216                         keycode 
== decimalPoint 
|| 
1228 #endif // wxUSE_TEXTCTRL 
1232 // ---------------------------------------------------------------------------- 
1233 // wxGridCellBoolEditor 
1234 // ---------------------------------------------------------------------------- 
1236 // the default values for GetValue() 
1237 wxString 
wxGridCellBoolEditor::ms_stringValues
[2] = { _T("1"), _T("") }; 
1239 void wxGridCellBoolEditor::Create(wxWindow
* parent
, 
1241                                   wxEvtHandler
* evtHandler
) 
1243     m_control 
= new wxCheckBox(parent
, id
, wxEmptyString
, 
1244                                wxDefaultPosition
, wxDefaultSize
, 
1247     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1250 void wxGridCellBoolEditor::SetSize(const wxRect
& r
) 
1252     bool resize 
= false; 
1253     wxSize size 
= m_control
->GetSize(); 
1254     wxCoord minSize 
= wxMin(r
.width
, r
.height
); 
1256     // check if the checkbox is not too big/small for this cell 
1257     wxSize sizeBest 
= m_control
->GetBestSize(); 
1258     if ( !(size 
== sizeBest
) ) 
1260         // reset to default size if it had been made smaller 
1266     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
1268         // leave 1 pixel margin 
1269         size
.x 
= size
.y 
= minSize 
- 2; 
1276         m_control
->SetSize(size
); 
1279     // position it in the centre of the rectangle (TODO: support alignment?) 
1281 #if defined(__WXGTK__) || defined (__WXMOTIF__) 
1282     // the checkbox without label still has some space to the right in wxGTK, 
1283     // so shift it to the right 
1285 #elif defined(__WXMSW__) 
1286     // here too, but in other way 
1291     int hAlign 
= wxALIGN_CENTRE
; 
1292     int vAlign 
= wxALIGN_CENTRE
; 
1294         GetCellAttr()->GetAlignment(& hAlign
, & vAlign
); 
1297     if (hAlign 
== wxALIGN_LEFT
) 
1305         y 
= r
.y 
+ r
.height 
/ 2 - size
.y 
/ 2; 
1307     else if (hAlign 
== wxALIGN_RIGHT
) 
1309         x 
= r
.x 
+ r
.width 
- size
.x 
- 2; 
1310         y 
= r
.y 
+ r
.height 
/ 2 - size
.y 
/ 2; 
1312     else if (hAlign 
== wxALIGN_CENTRE
) 
1314         x 
= r
.x 
+ r
.width 
/ 2 - size
.x 
/ 2; 
1315         y 
= r
.y 
+ r
.height 
/ 2 - size
.y 
/ 2; 
1318     m_control
->Move(x
, y
); 
1321 void wxGridCellBoolEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
1323     m_control
->Show(show
); 
1327         wxColour colBg 
= attr 
? attr
->GetBackgroundColour() : *wxLIGHT_GREY
; 
1328         CBox()->SetBackgroundColour(colBg
); 
1332 void wxGridCellBoolEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1334     wxASSERT_MSG(m_control
, 
1335                  wxT("The wxGridCellEditor must be created first!")); 
1337     if (grid
->GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
)) 
1339         m_startValue 
= grid
->GetTable()->GetValueAsBool(row
, col
); 
1343         wxString 
cellval( grid
->GetTable()->GetValue(row
, col
) ); 
1345         if ( cellval 
== ms_stringValues
[false] ) 
1346             m_startValue 
= false; 
1347         else if ( cellval 
== ms_stringValues
[true] ) 
1348             m_startValue 
= true; 
1351             // do not try to be smart here and convert it to true or false 
1352             // because we'll still overwrite it with something different and 
1353             // this risks to be very surprising for the user code, let them 
1355             wxFAIL_MSG( _T("invalid value for a cell with bool editor!") ); 
1359     CBox()->SetValue(m_startValue
); 
1363 bool wxGridCellBoolEditor::EndEdit(int row
, int col
, 
1366     wxASSERT_MSG(m_control
, 
1367                  wxT("The wxGridCellEditor must be created first!")); 
1369     bool changed 
= false; 
1370     bool value 
= CBox()->GetValue(); 
1371     if ( value 
!= m_startValue 
) 
1376         wxGridTableBase 
* const table 
= grid
->GetTable(); 
1377         if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
) ) 
1378             table
->SetValueAsBool(row
, col
, value
); 
1380             table
->SetValue(row
, col
, GetValue()); 
1386 void wxGridCellBoolEditor::Reset() 
1388     wxASSERT_MSG(m_control
, 
1389                  wxT("The wxGridCellEditor must be created first!")); 
1391     CBox()->SetValue(m_startValue
); 
1394 void wxGridCellBoolEditor::StartingClick() 
1396     CBox()->SetValue(!CBox()->GetValue()); 
1399 bool wxGridCellBoolEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1401     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1403         int keycode 
= event
.GetKeyCode(); 
1416 void wxGridCellBoolEditor::StartingKey(wxKeyEvent
& event
) 
1418     int keycode 
= event
.GetKeyCode(); 
1422             CBox()->SetValue(!CBox()->GetValue()); 
1426             CBox()->SetValue(true); 
1430             CBox()->SetValue(false); 
1435 wxString 
wxGridCellBoolEditor::GetValue() const 
1437   return ms_stringValues
[CBox()->GetValue()]; 
1441 wxGridCellBoolEditor::UseStringValues(const wxString
& valueTrue
, 
1442                                       const wxString
& valueFalse
) 
1444     ms_stringValues
[false] = valueFalse
; 
1445     ms_stringValues
[true] = valueTrue
; 
1449 wxGridCellBoolEditor::IsTrueValue(const wxString
& value
) 
1451     return value 
== ms_stringValues
[true]; 
1454 #endif // wxUSE_CHECKBOX 
1458 // ---------------------------------------------------------------------------- 
1459 // wxGridCellChoiceEditor 
1460 // ---------------------------------------------------------------------------- 
1462 wxGridCellChoiceEditor::wxGridCellChoiceEditor(const wxArrayString
& choices
, 
1464     : m_choices(choices
), 
1465       m_allowOthers(allowOthers
) { } 
1467 wxGridCellChoiceEditor::wxGridCellChoiceEditor(size_t count
, 
1468                                                const wxString choices
[], 
1470                       : m_allowOthers(allowOthers
) 
1474         m_choices
.Alloc(count
); 
1475         for ( size_t n 
= 0; n 
< count
; n
++ ) 
1477             m_choices
.Add(choices
[n
]); 
1482 wxGridCellEditor 
*wxGridCellChoiceEditor::Clone() const 
1484     wxGridCellChoiceEditor 
*editor 
= new wxGridCellChoiceEditor
; 
1485     editor
->m_allowOthers 
= m_allowOthers
; 
1486     editor
->m_choices 
= m_choices
; 
1491 void wxGridCellChoiceEditor::Create(wxWindow
* parent
, 
1493                                     wxEvtHandler
* evtHandler
) 
1495     m_control 
= new wxComboBox(parent
, id
, wxEmptyString
, 
1496                                wxDefaultPosition
, wxDefaultSize
, 
1498                                m_allowOthers 
? 0 : wxCB_READONLY
); 
1500     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1503 void wxGridCellChoiceEditor::PaintBackground(const wxRect
& rectCell
, 
1504                                              wxGridCellAttr 
* attr
) 
1506     // as we fill the entire client area, don't do anything here to minimize 
1509     // TODO: It doesn't actually fill the client area since the height of a 
1510     // combo always defaults to the standard.  Until someone has time to 
1511     // figure out the right rectangle to paint, just do it the normal way. 
1512     wxGridCellEditor::PaintBackground(rectCell
, attr
); 
1515 void wxGridCellChoiceEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1517     wxASSERT_MSG(m_control
, 
1518                  wxT("The wxGridCellEditor must be created first!")); 
1520     wxGridCellEditorEvtHandler
* evtHandler 
= NULL
; 
1522         evtHandler 
= wxDynamicCast(m_control
->GetEventHandler(), wxGridCellEditorEvtHandler
); 
1524     // Don't immediately end if we get a kill focus event within BeginEdit 
1526         evtHandler
->SetInSetFocus(true); 
1528     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
1532         Combo()->SetValue(m_startValue
); 
1536         // find the right position, or default to the first if not found 
1537         int pos 
= Combo()->FindString(m_startValue
); 
1538         if (pos 
== wxNOT_FOUND
) 
1540         Combo()->SetSelection(pos
); 
1543     Combo()->SetInsertionPointEnd(); 
1544     Combo()->SetFocus(); 
1548         // When dropping down the menu, a kill focus event 
1549         // happens after this point, so we can't reset the flag yet. 
1550 #if !defined(__WXGTK20__) 
1551         evtHandler
->SetInSetFocus(false); 
1556 bool wxGridCellChoiceEditor::EndEdit(int row
, int col
, 
1559     wxString value 
= Combo()->GetValue(); 
1560     if ( value 
== m_startValue 
) 
1563     grid
->GetTable()->SetValue(row
, col
, value
); 
1568 void wxGridCellChoiceEditor::Reset() 
1570     Combo()->SetValue(m_startValue
); 
1571     Combo()->SetInsertionPointEnd(); 
1574 void wxGridCellChoiceEditor::SetParameters(const wxString
& params
) 
1584     wxStringTokenizer 
tk(params
, _T(',')); 
1585     while ( tk
.HasMoreTokens() ) 
1587         m_choices
.Add(tk
.GetNextToken()); 
1591 // return the value in the text control 
1592 wxString 
wxGridCellChoiceEditor::GetValue() const 
1594   return Combo()->GetValue(); 
1597 #endif // wxUSE_COMBOBOX 
1599 // ---------------------------------------------------------------------------- 
1600 // wxGridCellEditorEvtHandler 
1601 // ---------------------------------------------------------------------------- 
1603 void wxGridCellEditorEvtHandler::OnKillFocus(wxFocusEvent
& event
) 
1605     // Don't disable the cell if we're just starting to edit it 
1610     m_grid
->DisableCellEditControl(); 
1615 void wxGridCellEditorEvtHandler::OnKeyDown(wxKeyEvent
& event
) 
1617     switch ( event
.GetKeyCode() ) 
1621             m_grid
->DisableCellEditControl(); 
1625             m_grid
->GetEventHandler()->ProcessEvent( event 
); 
1629         case WXK_NUMPAD_ENTER
: 
1630             if (!m_grid
->GetEventHandler()->ProcessEvent(event
)) 
1631                 m_editor
->HandleReturn(event
); 
1640 void wxGridCellEditorEvtHandler::OnChar(wxKeyEvent
& event
) 
1642     int row 
= m_grid
->GetGridCursorRow(); 
1643     int col 
= m_grid
->GetGridCursorCol(); 
1644     wxRect rect 
= m_grid
->CellToRect( row
, col 
); 
1646     m_grid
->GetGridWindow()->GetClientSize( &cw
, &ch 
); 
1648     // if cell width is smaller than grid client area, cell is wholly visible 
1649     bool wholeCellVisible 
= (rect
.GetWidth() < cw
); 
1651     switch ( event
.GetKeyCode() ) 
1656         case WXK_NUMPAD_ENTER
: 
1661             if ( wholeCellVisible 
) 
1663                 // no special processing needed... 
1668             // do special processing for partly visible cell... 
1670             // get the widths of all cells previous to this one 
1672             for ( int i 
= 0; i 
< col
; i
++ ) 
1674                 colXPos 
+= m_grid
->GetColSize(i
); 
1677             int xUnit 
= 1, yUnit 
= 1; 
1678             m_grid
->GetScrollPixelsPerUnit(&xUnit
, &yUnit
); 
1681                 m_grid
->Scroll(colXPos 
/ xUnit 
- 1, m_grid
->GetScrollPos(wxVERTICAL
)); 
1685                 m_grid
->Scroll(colXPos 
/ xUnit
, m_grid
->GetScrollPos(wxVERTICAL
)); 
1693             if ( wholeCellVisible 
) 
1695                 // no special processing needed... 
1700             // do special processing for partly visible cell... 
1703             wxString value 
= m_grid
->GetCellValue(row
, col
); 
1704             if ( wxEmptyString 
!= value 
) 
1706                 // get width of cell CONTENTS (text) 
1708                 wxFont font 
= m_grid
->GetCellFont(row
, col
); 
1709                 m_grid
->GetTextExtent(value
, &textWidth
, &y
, NULL
, NULL
, &font
); 
1711                 // try to RIGHT align the text by scrolling 
1712                 int client_right 
= m_grid
->GetGridWindow()->GetClientSize().GetWidth(); 
1714                 // (m_grid->GetScrollLineX()*2) is a factor for not scrolling to far, 
1715                 // otherwise the last part of the cell content might be hidden below the scroll bar 
1716                 // FIXME: maybe there is a more suitable correction? 
1717                 textWidth 
-= (client_right 
- (m_grid
->GetScrollLineX() * 2)); 
1718                 if ( textWidth 
< 0 ) 
1724             // get the widths of all cells previous to this one 
1726             for ( int i 
= 0; i 
< col
; i
++ ) 
1728                 colXPos 
+= m_grid
->GetColSize(i
); 
1731             // and add the (modified) text width of the cell contents 
1732             // as we'd like to see the last part of the cell contents 
1733             colXPos 
+= textWidth
; 
1735             int xUnit 
= 1, yUnit 
= 1; 
1736             m_grid
->GetScrollPixelsPerUnit(&xUnit
, &yUnit
); 
1737             m_grid
->Scroll(colXPos 
/ xUnit 
- 1, m_grid
->GetScrollPos(wxVERTICAL
)); 
1748 // ---------------------------------------------------------------------------- 
1749 // wxGridCellWorker is an (almost) empty common base class for 
1750 // wxGridCellRenderer and wxGridCellEditor managing ref counting 
1751 // ---------------------------------------------------------------------------- 
1753 void wxGridCellWorker::SetParameters(const wxString
& WXUNUSED(params
)) 
1758 wxGridCellWorker::~wxGridCellWorker() 
1762 // ============================================================================ 
1764 // ============================================================================ 
1766 // ---------------------------------------------------------------------------- 
1767 // wxGridCellRenderer 
1768 // ---------------------------------------------------------------------------- 
1770 void wxGridCellRenderer::Draw(wxGrid
& grid
, 
1771                               wxGridCellAttr
& attr
, 
1774                               int WXUNUSED(row
), int WXUNUSED(col
), 
1777     dc
.SetBackgroundMode( wxSOLID 
); 
1779     // grey out fields if the grid is disabled 
1780     if ( grid
.IsEnabled() ) 
1784             dc
.SetBrush( wxBrush(grid
.GetSelectionBackground(), wxSOLID
) ); 
1788             dc
.SetBrush( wxBrush(attr
.GetBackgroundColour(), wxSOLID
) ); 
1793         dc
.SetBrush(wxBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE
), wxSOLID
)); 
1796     dc
.SetPen( *wxTRANSPARENT_PEN 
); 
1797     dc
.DrawRectangle(rect
); 
1800 // ---------------------------------------------------------------------------- 
1801 // wxGridCellStringRenderer 
1802 // ---------------------------------------------------------------------------- 
1804 void wxGridCellStringRenderer::SetTextColoursAndFont(const wxGrid
& grid
, 
1805                                                      const wxGridCellAttr
& attr
, 
1809     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
1811     // TODO some special colours for attr.IsReadOnly() case? 
1813     // different coloured text when the grid is disabled 
1814     if ( grid
.IsEnabled() ) 
1818             dc
.SetTextBackground( grid
.GetSelectionBackground() ); 
1819             dc
.SetTextForeground( grid
.GetSelectionForeground() ); 
1823             dc
.SetTextBackground( attr
.GetBackgroundColour() ); 
1824             dc
.SetTextForeground( attr
.GetTextColour() ); 
1829         dc
.SetTextBackground(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE
)); 
1830         dc
.SetTextForeground(wxSystemSettings::GetColour(wxSYS_COLOUR_GRAYTEXT
)); 
1833     dc
.SetFont( attr
.GetFont() ); 
1836 wxSize 
wxGridCellStringRenderer::DoGetBestSize(const wxGridCellAttr
& attr
, 
1838                                                const wxString
& text
) 
1840     wxCoord x 
= 0, y 
= 0, max_x 
= 0; 
1841     dc
.SetFont(attr
.GetFont()); 
1842     wxStringTokenizer 
tk(text
, _T('\n')); 
1843     while ( tk
.HasMoreTokens() ) 
1845         dc
.GetTextExtent(tk
.GetNextToken(), &x
, &y
); 
1846         max_x 
= wxMax(max_x
, x
); 
1849     y 
*= 1 + text
.Freq(wxT('\n')); // multiply by the number of lines. 
1851     return wxSize(max_x
, y
); 
1854 wxSize 
wxGridCellStringRenderer::GetBestSize(wxGrid
& grid
, 
1855                                              wxGridCellAttr
& attr
, 
1859     return DoGetBestSize(attr
, dc
, grid
.GetCellValue(row
, col
)); 
1862 void wxGridCellStringRenderer::Draw(wxGrid
& grid
, 
1863                                     wxGridCellAttr
& attr
, 
1865                                     const wxRect
& rectCell
, 
1869     wxRect rect 
= rectCell
; 
1872     // erase only this cells background, overflow cells should have been erased 
1873     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1876     attr
.GetAlignment(&hAlign
, &vAlign
); 
1878     int overflowCols 
= 0; 
1880     if (attr
.GetOverflow()) 
1882         int cols 
= grid
.GetNumberCols(); 
1883         int best_width 
= GetBestSize(grid
,attr
,dc
,row
,col
).GetWidth(); 
1884         int cell_rows
, cell_cols
; 
1885         attr
.GetSize( &cell_rows
, &cell_cols 
); // shouldn't get here if <= 0 
1886         if ((best_width 
> rectCell
.width
) && (col 
< cols
) && grid
.GetTable()) 
1888             int i
, c_cols
, c_rows
; 
1889             for (i 
= col
+cell_cols
; i 
< cols
; i
++) 
1891                 bool is_empty 
= true; 
1892                 for (int j
=row
; j 
< row 
+ cell_rows
; j
++) 
1894                     // check w/ anchor cell for multicell block 
1895                     grid
.GetCellSize(j
, i
, &c_rows
, &c_cols
); 
1898                     if (!grid
.GetTable()->IsEmptyCell(j 
+ c_rows
, i
)) 
1907                     rect
.width 
+= grid
.GetColSize(i
); 
1915                 if (rect
.width 
>= best_width
) 
1919             overflowCols 
= i 
- col 
- cell_cols 
+ 1; 
1920             if (overflowCols 
>= cols
) 
1921                 overflowCols 
= cols 
- 1; 
1924         if (overflowCols 
> 0) // redraw overflow cells w/ proper hilight 
1926             hAlign 
= wxALIGN_LEFT
; // if oveflowed then it's left aligned 
1928             clip
.x 
+= rectCell
.width
; 
1929             // draw each overflow cell individually 
1930             int col_end 
= col 
+ cell_cols 
+ overflowCols
; 
1931             if (col_end 
>= grid
.GetNumberCols()) 
1932                 col_end 
= grid
.GetNumberCols() - 1; 
1933             for (int i 
= col 
+ cell_cols
; i 
<= col_end
; i
++) 
1935                 clip
.width 
= grid
.GetColSize(i
) - 1; 
1936                 dc
.DestroyClippingRegion(); 
1937                 dc
.SetClippingRegion(clip
); 
1939                 SetTextColoursAndFont(grid
, attr
, dc
, 
1940                         grid
.IsInSelection(row
,i
)); 
1942                 grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1943                         rect
, hAlign
, vAlign
); 
1944                 clip
.x 
+= grid
.GetColSize(i
) - 1; 
1950             dc
.DestroyClippingRegion(); 
1954     // now we only have to draw the text 
1955     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1957     grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1958                            rect
, hAlign
, vAlign
); 
1961 // ---------------------------------------------------------------------------- 
1962 // wxGridCellNumberRenderer 
1963 // ---------------------------------------------------------------------------- 
1965 wxString 
wxGridCellNumberRenderer::GetString(const wxGrid
& grid
, int row
, int col
) 
1967     wxGridTableBase 
*table 
= grid
.GetTable(); 
1969     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
1971         text
.Printf(_T("%ld"), table
->GetValueAsLong(row
, col
)); 
1975         text 
= table
->GetValue(row
, col
); 
1981 void wxGridCellNumberRenderer::Draw(wxGrid
& grid
, 
1982                                     wxGridCellAttr
& attr
, 
1984                                     const wxRect
& rectCell
, 
1988     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1990     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1992     // draw the text right aligned by default 
1994     attr
.GetAlignment(&hAlign
, &vAlign
); 
1995     hAlign 
= wxALIGN_RIGHT
; 
1997     wxRect rect 
= rectCell
; 
2000     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
2003 wxSize 
wxGridCellNumberRenderer::GetBestSize(wxGrid
& grid
, 
2004                                              wxGridCellAttr
& attr
, 
2008     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
2011 // ---------------------------------------------------------------------------- 
2012 // wxGridCellFloatRenderer 
2013 // ---------------------------------------------------------------------------- 
2015 wxGridCellFloatRenderer::wxGridCellFloatRenderer(int width
, int precision
) 
2018     SetPrecision(precision
); 
2021 wxGridCellRenderer 
*wxGridCellFloatRenderer::Clone() const 
2023     wxGridCellFloatRenderer 
*renderer 
= new wxGridCellFloatRenderer
; 
2024     renderer
->m_width 
= m_width
; 
2025     renderer
->m_precision 
= m_precision
; 
2026     renderer
->m_format 
= m_format
; 
2031 wxString 
wxGridCellFloatRenderer::GetString(const wxGrid
& grid
, int row
, int col
) 
2033     wxGridTableBase 
*table 
= grid
.GetTable(); 
2038     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
2040         val 
= table
->GetValueAsDouble(row
, col
); 
2045         text 
= table
->GetValue(row
, col
); 
2046         hasDouble 
= text
.ToDouble(&val
); 
2053             if ( m_width 
== -1 ) 
2055                 if ( m_precision 
== -1 ) 
2057                     // default width/precision 
2058                     m_format 
= _T("%f"); 
2062                     m_format
.Printf(_T("%%.%df"), m_precision
); 
2065             else if ( m_precision 
== -1 ) 
2067                 // default precision 
2068                 m_format
.Printf(_T("%%%d.f"), m_width
); 
2072                 m_format
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
2076         text
.Printf(m_format
, val
); 
2079     //else: text already contains the string 
2084 void wxGridCellFloatRenderer::Draw(wxGrid
& grid
, 
2085                                    wxGridCellAttr
& attr
, 
2087                                    const wxRect
& rectCell
, 
2091     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
2093     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
2095     // draw the text right aligned by default 
2097     attr
.GetAlignment(&hAlign
, &vAlign
); 
2098     hAlign 
= wxALIGN_RIGHT
; 
2100     wxRect rect 
= rectCell
; 
2103     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
2106 wxSize 
wxGridCellFloatRenderer::GetBestSize(wxGrid
& grid
, 
2107                                             wxGridCellAttr
& attr
, 
2111     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
2114 void wxGridCellFloatRenderer::SetParameters(const wxString
& params
) 
2118         // reset to defaults 
2124         wxString tmp 
= params
.BeforeFirst(_T(',')); 
2128             if ( tmp
.ToLong(&width
) ) 
2130                 SetWidth((int)width
); 
2134                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer width parameter string '%s ignored"), params
.c_str()); 
2138         tmp 
= params
.AfterFirst(_T(',')); 
2142             if ( tmp
.ToLong(&precision
) ) 
2144                 SetPrecision((int)precision
); 
2148                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer precision parameter string '%s ignored"), params
.c_str()); 
2154 // ---------------------------------------------------------------------------- 
2155 // wxGridCellBoolRenderer 
2156 // ---------------------------------------------------------------------------- 
2158 wxSize 
wxGridCellBoolRenderer::ms_sizeCheckMark
; 
2160 // FIXME these checkbox size calculations are really ugly... 
2162 // between checkmark and box 
2163 static const wxCoord wxGRID_CHECKMARK_MARGIN 
= 2; 
2165 wxSize 
wxGridCellBoolRenderer::GetBestSize(wxGrid
& grid
, 
2166                                            wxGridCellAttr
& WXUNUSED(attr
), 
2171     // compute it only once (no locks for MT safeness in GUI thread...) 
2172     if ( !ms_sizeCheckMark
.x 
) 
2174         // get checkbox size 
2175         wxCheckBox 
*checkbox 
= new wxCheckBox(&grid
, wxID_ANY
, wxEmptyString
); 
2176         wxSize size 
= checkbox
->GetBestSize(); 
2177         wxCoord checkSize 
= size
.y 
+ 2 * wxGRID_CHECKMARK_MARGIN
; 
2179         // FIXME wxGTK::wxCheckBox::GetBestSize() gives "wrong" result 
2180 #if defined(__WXGTK__) || defined(__WXMOTIF__) 
2181         checkSize 
-= size
.y 
/ 2; 
2186         ms_sizeCheckMark
.x 
= ms_sizeCheckMark
.y 
= checkSize
; 
2189     return ms_sizeCheckMark
; 
2192 void wxGridCellBoolRenderer::Draw(wxGrid
& grid
, 
2193                                   wxGridCellAttr
& attr
, 
2199     wxGridCellRenderer::Draw(grid
, attr
, dc
, rect
, row
, col
, isSelected
); 
2201     // draw a check mark in the centre (ignoring alignment - TODO) 
2202     wxSize size 
= GetBestSize(grid
, attr
, dc
, row
, col
); 
2204     // don't draw outside the cell 
2205     wxCoord minSize 
= wxMin(rect
.width
, rect
.height
); 
2206     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
2208         // and even leave (at least) 1 pixel margin 
2209         size
.x 
= size
.y 
= minSize 
- 2; 
2212     // draw a border around checkmark 
2214     attr
.GetAlignment(&hAlign
, &vAlign
); 
2217     if (hAlign 
== wxALIGN_CENTRE
) 
2219         rectBorder
.x 
= rect
.x 
+ rect
.width 
/ 2 - size
.x 
/ 2; 
2220         rectBorder
.y 
= rect
.y 
+ rect
.height 
/ 2 - size
.y 
/ 2; 
2221         rectBorder
.width 
= size
.x
; 
2222         rectBorder
.height 
= size
.y
; 
2224     else if (hAlign 
== wxALIGN_LEFT
) 
2226         rectBorder
.x 
= rect
.x 
+ 2; 
2227         rectBorder
.y 
= rect
.y 
+ rect
.height 
/ 2 - size
.y 
/ 2; 
2228         rectBorder
.width 
= size
.x
; 
2229         rectBorder
.height 
= size
.y
; 
2231     else if (hAlign 
== wxALIGN_RIGHT
) 
2233         rectBorder
.x 
= rect
.x 
+ rect
.width 
- size
.x 
- 2; 
2234         rectBorder
.y 
= rect
.y 
+ rect
.height 
/ 2 - size
.y 
/ 2; 
2235         rectBorder
.width 
= size
.x
; 
2236         rectBorder
.height 
= size
.y
; 
2240     if ( grid
.GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
) ) 
2242         value 
= grid
.GetTable()->GetValueAsBool(row
, col
); 
2246         wxString 
cellval( grid
.GetTable()->GetValue(row
, col
) ); 
2247         value 
= wxGridCellBoolEditor::IsTrueValue(cellval
); 
2252         wxRect rectMark 
= rectBorder
; 
2255         // MSW DrawCheckMark() is weird (and should probably be changed...) 
2256         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN 
/ 2); 
2260         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN
); 
2263         dc
.SetTextForeground(attr
.GetTextColour()); 
2264         dc
.DrawCheckMark(rectMark
); 
2267     dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
2268     dc
.SetPen(wxPen(attr
.GetTextColour(), 1, wxSOLID
)); 
2269     dc
.DrawRectangle(rectBorder
); 
2272 // ---------------------------------------------------------------------------- 
2274 // ---------------------------------------------------------------------------- 
2276 void wxGridCellAttr::Init(wxGridCellAttr 
*attrDefault
) 
2280     m_isReadOnly 
= Unset
; 
2285     m_attrkind 
= wxGridCellAttr::Cell
; 
2287     m_sizeRows 
= m_sizeCols 
= 1; 
2288     m_overflow 
= UnsetOverflow
; 
2290     SetDefAttr(attrDefault
); 
2293 wxGridCellAttr 
*wxGridCellAttr::Clone() const 
2295     wxGridCellAttr 
*attr 
= new wxGridCellAttr(m_defGridAttr
); 
2297     if ( HasTextColour() ) 
2298         attr
->SetTextColour(GetTextColour()); 
2299     if ( HasBackgroundColour() ) 
2300         attr
->SetBackgroundColour(GetBackgroundColour()); 
2302         attr
->SetFont(GetFont()); 
2303     if ( HasAlignment() ) 
2304         attr
->SetAlignment(m_hAlign
, m_vAlign
); 
2306     attr
->SetSize( m_sizeRows
, m_sizeCols 
); 
2310         attr
->SetRenderer(m_renderer
); 
2311         m_renderer
->IncRef(); 
2315         attr
->SetEditor(m_editor
); 
2320         attr
->SetReadOnly(); 
2322     attr
->SetOverflow( m_overflow 
== Overflow 
); 
2323     attr
->SetKind( m_attrkind 
); 
2328 void wxGridCellAttr::MergeWith(wxGridCellAttr 
*mergefrom
) 
2330     if ( !HasTextColour() && mergefrom
->HasTextColour() ) 
2331         SetTextColour(mergefrom
->GetTextColour()); 
2332     if ( !HasBackgroundColour() && mergefrom
->HasBackgroundColour() ) 
2333         SetBackgroundColour(mergefrom
->GetBackgroundColour()); 
2334     if ( !HasFont() && mergefrom
->HasFont() ) 
2335         SetFont(mergefrom
->GetFont()); 
2336     if ( !HasAlignment() && mergefrom
->HasAlignment() ) 
2339         mergefrom
->GetAlignment( &hAlign
, &vAlign
); 
2340         SetAlignment(hAlign
, vAlign
); 
2342     if ( !HasSize() && mergefrom
->HasSize() ) 
2343         mergefrom
->GetSize( &m_sizeRows
, &m_sizeCols 
); 
2345     // Directly access member functions as GetRender/Editor don't just return 
2346     // m_renderer/m_editor 
2348     // Maybe add support for merge of Render and Editor? 
2349     if (!HasRenderer() && mergefrom
->HasRenderer() ) 
2351         m_renderer 
= mergefrom
->m_renderer
; 
2352         m_renderer
->IncRef(); 
2354     if ( !HasEditor() && mergefrom
->HasEditor() ) 
2356         m_editor 
=  mergefrom
->m_editor
; 
2359     if ( !HasReadWriteMode() && mergefrom
->HasReadWriteMode() ) 
2360         SetReadOnly(mergefrom
->IsReadOnly()); 
2362     if (!HasOverflowMode() && mergefrom
->HasOverflowMode() ) 
2363         SetOverflow(mergefrom
->GetOverflow()); 
2365     SetDefAttr(mergefrom
->m_defGridAttr
); 
2368 void wxGridCellAttr::SetSize(int num_rows
, int num_cols
) 
2370     // The size of a cell is normally 1,1 
2372     // If this cell is larger (2,2) then this is the top left cell 
2373     // the other cells that will be covered (lower right cells) must be 
2374     // set to negative or zero values such that 
2375     // row + num_rows of the covered cell points to the larger cell (this cell) 
2376     // same goes for the col + num_cols. 
2378     // Size of 0,0 is NOT valid, neither is <=0 and any positive value 
2380     wxASSERT_MSG( (!((num_rows 
> 0) && (num_cols 
<= 0)) || 
2381                   !((num_rows 
<= 0) && (num_cols 
> 0)) || 
2382                   !((num_rows 
== 0) && (num_cols 
== 0))), 
2383                   wxT("wxGridCellAttr::SetSize only takes two postive values or negative/zero values")); 
2385     m_sizeRows 
= num_rows
; 
2386     m_sizeCols 
= num_cols
; 
2389 const wxColour
& wxGridCellAttr::GetTextColour() const 
2391     if (HasTextColour()) 
2395     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2397         return m_defGridAttr
->GetTextColour(); 
2401         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2402         return wxNullColour
; 
2406 const wxColour
& wxGridCellAttr::GetBackgroundColour() const 
2408     if (HasBackgroundColour()) 
2412     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2414         return m_defGridAttr
->GetBackgroundColour(); 
2418         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2419         return wxNullColour
; 
2423 const wxFont
& wxGridCellAttr::GetFont() const 
2429     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2431         return m_defGridAttr
->GetFont(); 
2435         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2440 void wxGridCellAttr::GetAlignment(int *hAlign
, int *vAlign
) const 
2449     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2451         m_defGridAttr
->GetAlignment(hAlign
, vAlign
); 
2455         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2459 void wxGridCellAttr::GetSize( int *num_rows
, int *num_cols 
) const 
2462         *num_rows 
= m_sizeRows
; 
2464         *num_cols 
= m_sizeCols
; 
2467 // GetRenderer and GetEditor use a slightly different decision path about 
2468 // which attribute to use.  If a non-default attr object has one then it is 
2469 // used, otherwise the default editor or renderer is fetched from the grid and 
2470 // used.  It should be the default for the data type of the cell.  If it is 
2471 // NULL (because the table has a type that the grid does not have in its 
2472 // registry), then the grid's default editor or renderer is used. 
2474 wxGridCellRenderer
* wxGridCellAttr::GetRenderer(wxGrid
* grid
, int row
, int col
) const 
2476     wxGridCellRenderer 
*renderer 
= NULL
; 
2478     if ( m_renderer 
&& this != m_defGridAttr 
) 
2480         // use the cells renderer if it has one 
2481         renderer 
= m_renderer
; 
2484     else // no non-default cell renderer 
2486         // get default renderer for the data type 
2489             // GetDefaultRendererForCell() will do IncRef() for us 
2490             renderer 
= grid
->GetDefaultRendererForCell(row
, col
); 
2493         if ( renderer 
== NULL 
) 
2495             if ( (m_defGridAttr 
!= NULL
) && (m_defGridAttr 
!= this) ) 
2497                 // if we still don't have one then use the grid default 
2498                 // (no need for IncRef() here neither) 
2499                 renderer 
= m_defGridAttr
->GetRenderer(NULL
, 0, 0); 
2501             else // default grid attr 
2503                 // use m_renderer which we had decided not to use initially 
2504                 renderer 
= m_renderer
; 
2511     // we're supposed to always find something 
2512     wxASSERT_MSG(renderer
, wxT("Missing default cell renderer")); 
2517 // same as above, except for s/renderer/editor/g 
2518 wxGridCellEditor
* wxGridCellAttr::GetEditor(wxGrid
* grid
, int row
, int col
) const 
2520     wxGridCellEditor 
*editor 
= NULL
; 
2522     if ( m_editor 
&& this != m_defGridAttr 
) 
2524         // use the cells editor if it has one 
2528     else // no non default cell editor 
2530         // get default editor for the data type 
2533             // GetDefaultEditorForCell() will do IncRef() for us 
2534             editor 
= grid
->GetDefaultEditorForCell(row
, col
); 
2537         if ( editor 
== NULL 
) 
2539             if ( (m_defGridAttr 
!= NULL
) && (m_defGridAttr 
!= this) ) 
2541                 // if we still don't have one then use the grid default 
2542                 // (no need for IncRef() here neither) 
2543                 editor 
= m_defGridAttr
->GetEditor(NULL
, 0, 0); 
2545             else // default grid attr 
2547                 // use m_editor which we had decided not to use initially 
2555     // we're supposed to always find something 
2556     wxASSERT_MSG(editor
, wxT("Missing default cell editor")); 
2561 // ---------------------------------------------------------------------------- 
2562 // wxGridCellAttrData 
2563 // ---------------------------------------------------------------------------- 
2565 void wxGridCellAttrData::SetAttr(wxGridCellAttr 
*attr
, int row
, int col
) 
2567     int n 
= FindIndex(row
, col
); 
2568     if ( n 
== wxNOT_FOUND 
) 
2570         // add the attribute 
2571         m_attrs
.Add(new wxGridCellWithAttr(row
, col
, attr
)); 
2575         // free the old attribute 
2576         m_attrs
[(size_t)n
].attr
->DecRef(); 
2580             // change the attribute 
2581             m_attrs
[(size_t)n
].attr 
= attr
; 
2585             // remove this attribute 
2586             m_attrs
.RemoveAt((size_t)n
); 
2591 wxGridCellAttr 
*wxGridCellAttrData::GetAttr(int row
, int col
) const 
2593     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2595     int n 
= FindIndex(row
, col
); 
2596     if ( n 
!= wxNOT_FOUND 
) 
2598         attr 
= m_attrs
[(size_t)n
].attr
; 
2605 void wxGridCellAttrData::UpdateAttrRows( size_t pos
, int numRows 
) 
2607     size_t count 
= m_attrs
.GetCount(); 
2608     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2610         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2611         wxCoord row 
= coords
.GetRow(); 
2612         if ((size_t)row 
>= pos
) 
2616                 // If rows inserted, include row counter where necessary 
2617                 coords
.SetRow(row 
+ numRows
); 
2619             else if (numRows 
< 0) 
2621                 // If rows deleted ... 
2622                 if ((size_t)row 
>= pos 
- numRows
) 
2624                     // ...either decrement row counter (if row still exists)... 
2625                     coords
.SetRow(row 
+ numRows
); 
2629                     // ...or remove the attribute 
2630                     // No need to DecRef the attribute itself since this is 
2631                     // done be wxGridCellWithAttr's destructor! 
2632                     m_attrs
.RemoveAt(n
); 
2641 void wxGridCellAttrData::UpdateAttrCols( size_t pos
, int numCols 
) 
2643     size_t count 
= m_attrs
.GetCount(); 
2644     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2646         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2647         wxCoord col 
= coords
.GetCol(); 
2648         if ( (size_t)col 
>= pos 
) 
2652                 // If rows inserted, include row counter where necessary 
2653                 coords
.SetCol(col 
+ numCols
); 
2655             else if (numCols 
< 0) 
2657                 // If rows deleted ... 
2658                 if ((size_t)col 
>= pos 
- numCols
) 
2660                     // ...either decrement row counter (if row still exists)... 
2661                     coords
.SetCol(col 
+ numCols
); 
2665                     // ...or remove the attribute 
2666                     // No need to DecRef the attribute itself since this is 
2667                     // done be wxGridCellWithAttr's destructor! 
2668                     m_attrs
.RemoveAt(n
); 
2677 int wxGridCellAttrData::FindIndex(int row
, int col
) const 
2679     size_t count 
= m_attrs
.GetCount(); 
2680     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2682         const wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2683         if ( (coords
.GetRow() == row
) && (coords
.GetCol() == col
) ) 
2692 // ---------------------------------------------------------------------------- 
2693 // wxGridRowOrColAttrData 
2694 // ---------------------------------------------------------------------------- 
2696 wxGridRowOrColAttrData::~wxGridRowOrColAttrData() 
2698     size_t count 
= m_attrs
.Count(); 
2699     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2701         m_attrs
[n
]->DecRef(); 
2705 wxGridCellAttr 
*wxGridRowOrColAttrData::GetAttr(int rowOrCol
) const 
2707     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2709     int n 
= m_rowsOrCols
.Index(rowOrCol
); 
2710     if ( n 
!= wxNOT_FOUND 
) 
2712         attr 
= m_attrs
[(size_t)n
]; 
2719 void wxGridRowOrColAttrData::SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
) 
2721     int i 
= m_rowsOrCols
.Index(rowOrCol
); 
2722     if ( i 
== wxNOT_FOUND 
) 
2724         // add the attribute 
2725         m_rowsOrCols
.Add(rowOrCol
); 
2730         size_t n 
= (size_t)i
; 
2733             // change the attribute 
2734             m_attrs
[n
]->DecRef(); 
2739             // remove this attribute 
2740             m_attrs
[n
]->DecRef(); 
2741             m_rowsOrCols
.RemoveAt(n
); 
2742             m_attrs
.RemoveAt(n
); 
2747 void wxGridRowOrColAttrData::UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
) 
2749     size_t count 
= m_attrs
.GetCount(); 
2750     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2752         int & rowOrCol 
= m_rowsOrCols
[n
]; 
2753         if ( (size_t)rowOrCol 
>= pos 
) 
2755             if ( numRowsOrCols 
> 0 ) 
2757                 // If rows inserted, include row counter where necessary 
2758                 rowOrCol 
+= numRowsOrCols
; 
2760             else if ( numRowsOrCols 
< 0) 
2762                 // If rows deleted, either decrement row counter (if row still exists) 
2763                 if ((size_t)rowOrCol 
>= pos 
- numRowsOrCols
) 
2764                     rowOrCol 
+= numRowsOrCols
; 
2767                     m_rowsOrCols
.RemoveAt(n
); 
2768                     m_attrs
[n
]->DecRef(); 
2769                     m_attrs
.RemoveAt(n
); 
2778 // ---------------------------------------------------------------------------- 
2779 // wxGridCellAttrProvider 
2780 // ---------------------------------------------------------------------------- 
2782 wxGridCellAttrProvider::wxGridCellAttrProvider() 
2784     m_data 
= (wxGridCellAttrProviderData 
*)NULL
; 
2787 wxGridCellAttrProvider::~wxGridCellAttrProvider() 
2792 void wxGridCellAttrProvider::InitData() 
2794     m_data 
= new wxGridCellAttrProviderData
; 
2797 wxGridCellAttr 
*wxGridCellAttrProvider::GetAttr(int row
, int col
, 
2798                                                 wxGridCellAttr::wxAttrKind  kind 
) const 
2800     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2805             case (wxGridCellAttr::Any
): 
2806                 // Get cached merge attributes. 
2807                 // Currently not used as no cache implemented as not mutable 
2808                 // attr = m_data->m_mergeAttr.GetAttr(row, col); 
2811                     // Basically implement old version. 
2812                     // Also check merge cache, so we don't have to re-merge every time.. 
2813                     wxGridCellAttr 
*attrcell 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2814                     wxGridCellAttr 
*attrrow 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2815                     wxGridCellAttr 
*attrcol 
= m_data
->m_colAttrs
.GetAttr(col
); 
2817                     if ((attrcell 
!= attrrow
) && (attrrow 
!= attrcol
) && (attrcell 
!= attrcol
)) 
2819                         // Two or more are non NULL 
2820                         attr 
= new wxGridCellAttr
; 
2821                         attr
->SetKind(wxGridCellAttr::Merged
); 
2823                         // Order is important.. 
2826                             attr
->MergeWith(attrcell
); 
2831                             attr
->MergeWith(attrcol
); 
2836                             attr
->MergeWith(attrrow
); 
2840                         // store merge attr if cache implemented 
2842                         //m_data->m_mergeAttr.SetAttr(attr, row, col); 
2846                         // one or none is non null return it or null. 
2865             case (wxGridCellAttr::Cell
): 
2866                 attr 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2869             case (wxGridCellAttr::Col
): 
2870                 attr 
= m_data
->m_colAttrs
.GetAttr(col
); 
2873             case (wxGridCellAttr::Row
): 
2874                 attr 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2879                 // (wxGridCellAttr::Default): 
2880                 // (wxGridCellAttr::Merged): 
2888 void wxGridCellAttrProvider::SetAttr(wxGridCellAttr 
*attr
, 
2894     m_data
->m_cellAttrs
.SetAttr(attr
, row
, col
); 
2897 void wxGridCellAttrProvider::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
2902     m_data
->m_rowAttrs
.SetAttr(attr
, row
); 
2905 void wxGridCellAttrProvider::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
2910     m_data
->m_colAttrs
.SetAttr(attr
, col
); 
2913 void wxGridCellAttrProvider::UpdateAttrRows( size_t pos
, int numRows 
) 
2917         m_data
->m_cellAttrs
.UpdateAttrRows( pos
, numRows 
); 
2919         m_data
->m_rowAttrs
.UpdateAttrRowsOrCols( pos
, numRows 
); 
2923 void wxGridCellAttrProvider::UpdateAttrCols( size_t pos
, int numCols 
) 
2927         m_data
->m_cellAttrs
.UpdateAttrCols( pos
, numCols 
); 
2929         m_data
->m_colAttrs
.UpdateAttrRowsOrCols( pos
, numCols 
); 
2933 // ---------------------------------------------------------------------------- 
2934 // wxGridTypeRegistry 
2935 // ---------------------------------------------------------------------------- 
2937 wxGridTypeRegistry::~wxGridTypeRegistry() 
2939     size_t count 
= m_typeinfo
.Count(); 
2940     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2941         delete m_typeinfo
[i
]; 
2944 void wxGridTypeRegistry::RegisterDataType(const wxString
& typeName
, 
2945                                           wxGridCellRenderer
* renderer
, 
2946                                           wxGridCellEditor
* editor
) 
2948     wxGridDataTypeInfo
* info 
= new wxGridDataTypeInfo(typeName
, renderer
, editor
); 
2950     // is it already registered? 
2951     int loc 
= FindRegisteredDataType(typeName
); 
2952     if ( loc 
!= wxNOT_FOUND 
) 
2954         delete m_typeinfo
[loc
]; 
2955         m_typeinfo
[loc
] = info
; 
2959         m_typeinfo
.Add(info
); 
2963 int wxGridTypeRegistry::FindRegisteredDataType(const wxString
& typeName
) 
2965     size_t count 
= m_typeinfo
.GetCount(); 
2966     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2968         if ( typeName 
== m_typeinfo
[i
]->m_typeName 
) 
2977 int wxGridTypeRegistry::FindDataType(const wxString
& typeName
) 
2979     int index 
= FindRegisteredDataType(typeName
); 
2980     if ( index 
== wxNOT_FOUND 
) 
2982         // check whether this is one of the standard ones, in which case 
2983         // register it "on the fly" 
2985         if ( typeName 
== wxGRID_VALUE_STRING 
) 
2987             RegisterDataType(wxGRID_VALUE_STRING
, 
2988                              new wxGridCellStringRenderer
, 
2989                              new wxGridCellTextEditor
); 
2992 #endif // wxUSE_TEXTCTRL 
2994         if ( typeName 
== wxGRID_VALUE_BOOL 
) 
2996             RegisterDataType(wxGRID_VALUE_BOOL
, 
2997                              new wxGridCellBoolRenderer
, 
2998                              new wxGridCellBoolEditor
); 
3001 #endif // wxUSE_CHECKBOX 
3003         if ( typeName 
== wxGRID_VALUE_NUMBER 
) 
3005             RegisterDataType(wxGRID_VALUE_NUMBER
, 
3006                              new wxGridCellNumberRenderer
, 
3007                              new wxGridCellNumberEditor
); 
3009         else if ( typeName 
== wxGRID_VALUE_FLOAT 
) 
3011             RegisterDataType(wxGRID_VALUE_FLOAT
, 
3012                              new wxGridCellFloatRenderer
, 
3013                              new wxGridCellFloatEditor
); 
3016 #endif // wxUSE_TEXTCTRL 
3018         if ( typeName 
== wxGRID_VALUE_CHOICE 
) 
3020             RegisterDataType(wxGRID_VALUE_CHOICE
, 
3021                              new wxGridCellStringRenderer
, 
3022                              new wxGridCellChoiceEditor
); 
3025 #endif // wxUSE_COMBOBOX 
3030         // we get here only if just added the entry for this type, so return 
3032         index 
= m_typeinfo
.GetCount() - 1; 
3038 int wxGridTypeRegistry::FindOrCloneDataType(const wxString
& typeName
) 
3040     int index 
= FindDataType(typeName
); 
3041     if ( index 
== wxNOT_FOUND 
) 
3043         // the first part of the typename is the "real" type, anything after ':' 
3044         // are the parameters for the renderer 
3045         index 
= FindDataType(typeName
.BeforeFirst(_T(':'))); 
3046         if ( index 
== wxNOT_FOUND 
) 
3051         wxGridCellRenderer 
*renderer 
= GetRenderer(index
); 
3052         wxGridCellRenderer 
*rendererOld 
= renderer
; 
3053         renderer 
= renderer
->Clone(); 
3054         rendererOld
->DecRef(); 
3056         wxGridCellEditor 
*editor 
= GetEditor(index
); 
3057         wxGridCellEditor 
*editorOld 
= editor
; 
3058         editor 
= editor
->Clone(); 
3059         editorOld
->DecRef(); 
3061         // do it even if there are no parameters to reset them to defaults 
3062         wxString params 
= typeName
.AfterFirst(_T(':')); 
3063         renderer
->SetParameters(params
); 
3064         editor
->SetParameters(params
); 
3066         // register the new typename 
3067         RegisterDataType(typeName
, renderer
, editor
); 
3069         // we just registered it, it's the last one 
3070         index 
= m_typeinfo
.GetCount() - 1; 
3076 wxGridCellRenderer
* wxGridTypeRegistry::GetRenderer(int index
) 
3078     wxGridCellRenderer
* renderer 
= m_typeinfo
[index
]->m_renderer
; 
3085 wxGridCellEditor
* wxGridTypeRegistry::GetEditor(int index
) 
3087     wxGridCellEditor
* editor 
= m_typeinfo
[index
]->m_editor
; 
3094 // ---------------------------------------------------------------------------- 
3096 // ---------------------------------------------------------------------------- 
3098 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase
, wxObject 
) 
3100 wxGridTableBase::wxGridTableBase() 
3102     m_view 
= (wxGrid 
*) NULL
; 
3103     m_attrProvider 
= (wxGridCellAttrProvider 
*) NULL
; 
3106 wxGridTableBase::~wxGridTableBase() 
3108     delete m_attrProvider
; 
3111 void wxGridTableBase::SetAttrProvider(wxGridCellAttrProvider 
*attrProvider
) 
3113     delete m_attrProvider
; 
3114     m_attrProvider 
= attrProvider
; 
3117 bool wxGridTableBase::CanHaveAttributes() 
3119     if ( ! GetAttrProvider() ) 
3121         // use the default attr provider by default 
3122         SetAttrProvider(new wxGridCellAttrProvider
); 
3128 wxGridCellAttr 
*wxGridTableBase::GetAttr(int row
, int col
, wxGridCellAttr::wxAttrKind  kind
) 
3130     if ( m_attrProvider 
) 
3131         return m_attrProvider
->GetAttr(row
, col
, kind
); 
3133         return (wxGridCellAttr 
*)NULL
; 
3136 void wxGridTableBase::SetAttr(wxGridCellAttr
* attr
, int row
, int col
) 
3138     if ( m_attrProvider 
) 
3140         attr
->SetKind(wxGridCellAttr::Cell
); 
3141         m_attrProvider
->SetAttr(attr
, row
, col
); 
3145         // as we take ownership of the pointer and don't store it, we must 
3151 void wxGridTableBase::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
3153     if ( m_attrProvider 
) 
3155         attr
->SetKind(wxGridCellAttr::Row
); 
3156         m_attrProvider
->SetRowAttr(attr
, row
); 
3160         // as we take ownership of the pointer and don't store it, we must 
3166 void wxGridTableBase::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
3168     if ( m_attrProvider 
) 
3170         attr
->SetKind(wxGridCellAttr::Col
); 
3171         m_attrProvider
->SetColAttr(attr
, col
); 
3175         // as we take ownership of the pointer and don't store it, we must 
3181 bool wxGridTableBase::InsertRows( size_t WXUNUSED(pos
), 
3182                                   size_t WXUNUSED(numRows
) ) 
3184     wxFAIL_MSG( wxT("Called grid table class function InsertRows\nbut your derived table class does not override this function") ); 
3189 bool wxGridTableBase::AppendRows( size_t WXUNUSED(numRows
) ) 
3191     wxFAIL_MSG( wxT("Called grid table class function AppendRows\nbut your derived table class does not override this function")); 
3196 bool wxGridTableBase::DeleteRows( size_t WXUNUSED(pos
), 
3197                                   size_t WXUNUSED(numRows
) ) 
3199     wxFAIL_MSG( wxT("Called grid table class function DeleteRows\nbut your derived table class does not override this function")); 
3204 bool wxGridTableBase::InsertCols( size_t WXUNUSED(pos
), 
3205                                   size_t WXUNUSED(numCols
) ) 
3207     wxFAIL_MSG( wxT("Called grid table class function InsertCols\nbut your derived table class does not override this function")); 
3212 bool wxGridTableBase::AppendCols( size_t WXUNUSED(numCols
) ) 
3214     wxFAIL_MSG(wxT("Called grid table class function AppendCols\nbut your derived table class does not override this function")); 
3219 bool wxGridTableBase::DeleteCols( size_t WXUNUSED(pos
), 
3220                                   size_t WXUNUSED(numCols
) ) 
3222     wxFAIL_MSG( wxT("Called grid table class function DeleteCols\nbut your derived table class does not override this function")); 
3227 wxString 
wxGridTableBase::GetRowLabelValue( int row 
) 
3231     // RD: Starting the rows at zero confuses users, 
3232     // no matter how much it makes sense to us geeks. 
3238 wxString 
wxGridTableBase::GetColLabelValue( int col 
) 
3240     // default col labels are: 
3241     //   cols 0 to 25   : A-Z 
3242     //   cols 26 to 675 : AA-ZZ 
3247     for ( n 
= 1; ; n
++ ) 
3249         s 
+= (wxChar
) (_T('A') + (wxChar
)(col 
% 26)); 
3255     // reverse the string... 
3257     for ( i 
= 0; i 
< n
; i
++ ) 
3265 wxString 
wxGridTableBase::GetTypeName( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3267     return wxGRID_VALUE_STRING
; 
3270 bool wxGridTableBase::CanGetValueAs( int WXUNUSED(row
), int WXUNUSED(col
), 
3271                                      const wxString
& typeName 
) 
3273     return typeName 
== wxGRID_VALUE_STRING
; 
3276 bool wxGridTableBase::CanSetValueAs( int row
, int col
, const wxString
& typeName 
) 
3278     return CanGetValueAs(row
, col
, typeName
); 
3281 long wxGridTableBase::GetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3286 double wxGridTableBase::GetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3291 bool wxGridTableBase::GetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3296 void wxGridTableBase::SetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
), 
3297                                       long WXUNUSED(value
) ) 
3301 void wxGridTableBase::SetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
), 
3302                                         double WXUNUSED(value
) ) 
3306 void wxGridTableBase::SetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
), 
3307                                       bool WXUNUSED(value
) ) 
3311 void* wxGridTableBase::GetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
3312                                          const wxString
& WXUNUSED(typeName
) ) 
3317 void  wxGridTableBase::SetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
3318                                          const wxString
& WXUNUSED(typeName
), 
3319                                          void* WXUNUSED(value
) ) 
3323 ////////////////////////////////////////////////////////////////////// 
3325 // Message class for the grid table to send requests and notifications 
3329 wxGridTableMessage::wxGridTableMessage() 
3331     m_table 
= (wxGridTableBase 
*) NULL
; 
3337 wxGridTableMessage::wxGridTableMessage( wxGridTableBase 
*table
, int id
, 
3338                                         int commandInt1
, int commandInt2 
) 
3342     m_comInt1 
= commandInt1
; 
3343     m_comInt2 
= commandInt2
; 
3346 ////////////////////////////////////////////////////////////////////// 
3348 // A basic grid table for string data. An object of this class will 
3349 // created by wxGrid if you don't specify an alternative table class. 
3352 WX_DEFINE_OBJARRAY(wxGridStringArray
) 
3354 IMPLEMENT_DYNAMIC_CLASS( wxGridStringTable
, wxGridTableBase 
) 
3356 wxGridStringTable::wxGridStringTable() 
3361 wxGridStringTable::wxGridStringTable( int numRows
, int numCols 
) 
3364     m_data
.Alloc( numRows 
); 
3367     sa
.Alloc( numCols 
); 
3368     sa
.Add( wxEmptyString
, numCols 
); 
3370     m_data
.Add( sa
, numRows 
); 
3373 wxGridStringTable::~wxGridStringTable() 
3377 int wxGridStringTable::GetNumberRows() 
3379     return m_data
.GetCount(); 
3382 int wxGridStringTable::GetNumberCols() 
3384     if ( m_data
.GetCount() > 0 ) 
3385         return m_data
[0].GetCount(); 
3390 wxString 
wxGridStringTable::GetValue( int row
, int col 
) 
3392     wxCHECK_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3394                  _T("invalid row or column index in wxGridStringTable") ); 
3396     return m_data
[row
][col
]; 
3399 void wxGridStringTable::SetValue( int row
, int col
, const wxString
& value 
) 
3401     wxCHECK_RET( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3402                  _T("invalid row or column index in wxGridStringTable") ); 
3404     m_data
[row
][col
] = value
; 
3407 bool wxGridStringTable::IsEmptyCell( int row
, int col 
) 
3409     wxCHECK_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3411                   _T("invalid row or column index in wxGridStringTable") ); 
3413     return (m_data
[row
][col
] == wxEmptyString
); 
3416 void wxGridStringTable::Clear() 
3419     int numRows
, numCols
; 
3421     numRows 
= m_data
.GetCount(); 
3424         numCols 
= m_data
[0].GetCount(); 
3426         for ( row 
= 0; row 
< numRows
; row
++ ) 
3428             for ( col 
= 0; col 
< numCols
; col
++ ) 
3430                 m_data
[row
][col
] = wxEmptyString
; 
3436 bool wxGridStringTable::InsertRows( size_t pos
, size_t numRows 
) 
3438     size_t curNumRows 
= m_data
.GetCount(); 
3439     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3440                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3442     if ( pos 
>= curNumRows 
) 
3444         return AppendRows( numRows 
); 
3448     sa
.Alloc( curNumCols 
); 
3449     sa
.Add( wxEmptyString
, curNumCols 
); 
3450     m_data
.Insert( sa
, pos
, numRows 
); 
3454         wxGridTableMessage 
msg( this, 
3455                                 wxGRIDTABLE_NOTIFY_ROWS_INSERTED
, 
3459         GetView()->ProcessTableMessage( msg 
); 
3465 bool wxGridStringTable::AppendRows( size_t numRows 
) 
3467     size_t curNumRows 
= m_data
.GetCount(); 
3468     size_t curNumCols 
= ( curNumRows 
> 0 
3469                          ? m_data
[0].GetCount() 
3470                          : ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3473     if ( curNumCols 
> 0 ) 
3475         sa
.Alloc( curNumCols 
); 
3476         sa
.Add( wxEmptyString
, curNumCols 
); 
3479     m_data
.Add( sa
, numRows 
); 
3483         wxGridTableMessage 
msg( this, 
3484                                 wxGRIDTABLE_NOTIFY_ROWS_APPENDED
, 
3487         GetView()->ProcessTableMessage( msg 
); 
3493 bool wxGridStringTable::DeleteRows( size_t pos
, size_t numRows 
) 
3495     size_t curNumRows 
= m_data
.GetCount(); 
3497     if ( pos 
>= curNumRows 
) 
3499         wxFAIL_MSG( wxString::Format
 
3501                         wxT("Called wxGridStringTable::DeleteRows(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu rows"), 
3503                         (unsigned long)numRows
, 
3504                         (unsigned long)curNumRows
 
3510     if ( numRows 
> curNumRows 
- pos 
) 
3512         numRows 
= curNumRows 
- pos
; 
3515     if ( numRows 
>= curNumRows 
) 
3521         m_data
.RemoveAt( pos
, numRows 
); 
3526         wxGridTableMessage 
msg( this, 
3527                                 wxGRIDTABLE_NOTIFY_ROWS_DELETED
, 
3531         GetView()->ProcessTableMessage( msg 
); 
3537 bool wxGridStringTable::InsertCols( size_t pos
, size_t numCols 
) 
3541     size_t curNumRows 
= m_data
.GetCount(); 
3542     size_t curNumCols 
= ( curNumRows 
> 0 
3543                          ? m_data
[0].GetCount() 
3544                          : ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3546     if ( pos 
>= curNumCols 
) 
3548         return AppendCols( numCols 
); 
3551     if ( !m_colLabels
.IsEmpty() ) 
3553         m_colLabels
.Insert( wxEmptyString
, pos
, numCols 
); 
3556         for ( i 
= pos
; i 
< pos 
+ numCols
; i
++ ) 
3557             m_colLabels
[i
] = wxGridTableBase::GetColLabelValue( i 
); 
3560     for ( row 
= 0; row 
< curNumRows
; row
++ ) 
3562         for ( col 
= pos
; col 
< pos 
+ numCols
; col
++ ) 
3564             m_data
[row
].Insert( wxEmptyString
, col 
); 
3570         wxGridTableMessage 
msg( this, 
3571                                 wxGRIDTABLE_NOTIFY_COLS_INSERTED
, 
3575         GetView()->ProcessTableMessage( msg 
); 
3581 bool wxGridStringTable::AppendCols( size_t numCols 
) 
3585     size_t curNumRows 
= m_data
.GetCount(); 
3590         // TODO: something better than this ? 
3592         wxFAIL_MSG( wxT("Unable to append cols to a grid table with no rows.\nCall AppendRows() first") ); 
3597     for ( row 
= 0; row 
< curNumRows
; row
++ ) 
3599         m_data
[row
].Add( wxEmptyString
, numCols 
); 
3604         wxGridTableMessage 
msg( this, 
3605                                 wxGRIDTABLE_NOTIFY_COLS_APPENDED
, 
3608         GetView()->ProcessTableMessage( msg 
); 
3614 bool wxGridStringTable::DeleteCols( size_t pos
, size_t numCols 
) 
3618     size_t curNumRows 
= m_data
.GetCount(); 
3619     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3620                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3622     if ( pos 
>= curNumCols 
) 
3624         wxFAIL_MSG( wxString::Format
 
3626                         wxT("Called wxGridStringTable::DeleteCols(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu cols"), 
3628                         (unsigned long)numCols
, 
3629                         (unsigned long)curNumCols
 
3636         colID 
= GetView()->GetColAt( pos 
); 
3640     if ( numCols 
> curNumCols 
- colID 
) 
3642         numCols 
= curNumCols 
- colID
; 
3645     if ( !m_colLabels
.IsEmpty() ) 
3647         m_colLabels
.RemoveAt( colID
, numCols 
); 
3650     for ( row 
= 0; row 
< curNumRows
; row
++ ) 
3652         if ( numCols 
>= curNumCols 
) 
3654             m_data
[row
].Clear(); 
3658             m_data
[row
].RemoveAt( colID
, numCols 
); 
3664         wxGridTableMessage 
msg( this, 
3665                                 wxGRIDTABLE_NOTIFY_COLS_DELETED
, 
3669         GetView()->ProcessTableMessage( msg 
); 
3675 wxString 
wxGridStringTable::GetRowLabelValue( int row 
) 
3677     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3679         // using default label 
3681         return wxGridTableBase::GetRowLabelValue( row 
); 
3685         return m_rowLabels
[row
]; 
3689 wxString 
wxGridStringTable::GetColLabelValue( int col 
) 
3691     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3693         // using default label 
3695         return wxGridTableBase::GetColLabelValue( col 
); 
3699         return m_colLabels
[col
]; 
3703 void wxGridStringTable::SetRowLabelValue( int row
, const wxString
& value 
) 
3705     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3707         int n 
= m_rowLabels
.GetCount(); 
3710         for ( i 
= n
; i 
<= row
; i
++ ) 
3712             m_rowLabels
.Add( wxGridTableBase::GetRowLabelValue(i
) ); 
3716     m_rowLabels
[row
] = value
; 
3719 void wxGridStringTable::SetColLabelValue( int col
, const wxString
& value 
) 
3721     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3723         int n 
= m_colLabels
.GetCount(); 
3726         for ( i 
= n
; i 
<= col
; i
++ ) 
3728             m_colLabels
.Add( wxGridTableBase::GetColLabelValue(i
) ); 
3732     m_colLabels
[col
] = value
; 
3736 ////////////////////////////////////////////////////////////////////// 
3737 ////////////////////////////////////////////////////////////////////// 
3739 IMPLEMENT_DYNAMIC_CLASS( wxGridRowLabelWindow
, wxWindow 
) 
3741 BEGIN_EVENT_TABLE( wxGridRowLabelWindow
, wxWindow 
) 
3742     EVT_PAINT( wxGridRowLabelWindow::OnPaint 
) 
3743     EVT_MOUSEWHEEL( wxGridRowLabelWindow::OnMouseWheel 
) 
3744     EVT_MOUSE_EVENTS( wxGridRowLabelWindow::OnMouseEvent 
) 
3745     EVT_KEY_DOWN( wxGridRowLabelWindow::OnKeyDown 
) 
3746     EVT_KEY_UP( wxGridRowLabelWindow::OnKeyUp 
) 
3747     EVT_CHAR( wxGridRowLabelWindow::OnChar 
) 
3750 wxGridRowLabelWindow::wxGridRowLabelWindow( wxGrid 
*parent
, 
3752                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3753   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
| wxBORDER_NONE 
| wxFULL_REPAINT_ON_RESIZE 
) 
3758 void wxGridRowLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3762     // NO - don't do this because it will set both the x and y origin 
3763     // coords to match the parent scrolled window and we just want to 
3764     // set the y coord  - MB 
3766     // m_owner->PrepareDC( dc ); 
3769     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3770     wxPoint pt 
= dc
.GetDeviceOrigin(); 
3771     dc
.SetDeviceOrigin( pt
.x
, pt
.y
-y 
); 
3773     wxArrayInt rows 
= m_owner
->CalcRowLabelsExposed( GetUpdateRegion() ); 
3774     m_owner
->DrawRowLabels( dc
, rows 
); 
3777 void wxGridRowLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3779     m_owner
->ProcessRowLabelMouseEvent( event 
); 
3782 void wxGridRowLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3784     m_owner
->GetEventHandler()->ProcessEvent( event 
); 
3787 // This seems to be required for wxMotif otherwise the mouse 
3788 // cursor must be in the cell edit control to get key events 
3790 void wxGridRowLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3792     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3796 void wxGridRowLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3798     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3802 void wxGridRowLabelWindow::OnChar( wxKeyEvent
& event 
) 
3804     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3808 ////////////////////////////////////////////////////////////////////// 
3810 IMPLEMENT_DYNAMIC_CLASS( wxGridColLabelWindow
, wxWindow 
) 
3812 BEGIN_EVENT_TABLE( wxGridColLabelWindow
, wxWindow 
) 
3813     EVT_PAINT( wxGridColLabelWindow::OnPaint 
) 
3814     EVT_MOUSEWHEEL( wxGridColLabelWindow::OnMouseWheel 
) 
3815     EVT_MOUSE_EVENTS( wxGridColLabelWindow::OnMouseEvent 
) 
3816     EVT_KEY_DOWN( wxGridColLabelWindow::OnKeyDown 
) 
3817     EVT_KEY_UP( wxGridColLabelWindow::OnKeyUp 
) 
3818     EVT_CHAR( wxGridColLabelWindow::OnChar 
) 
3821 wxGridColLabelWindow::wxGridColLabelWindow( wxGrid 
*parent
, 
3823                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3824   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
| wxBORDER_NONE 
| wxFULL_REPAINT_ON_RESIZE 
) 
3829 void wxGridColLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3833     // NO - don't do this because it will set both the x and y origin 
3834     // coords to match the parent scrolled window and we just want to 
3835     // set the x coord  - MB 
3837     // m_owner->PrepareDC( dc ); 
3840     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3841     wxPoint pt 
= dc
.GetDeviceOrigin(); 
3842     if (GetLayoutDirection() == wxLayout_RightToLeft
) 
3843         dc
.SetDeviceOrigin( pt
.x
+x
, pt
.y 
); 
3845         dc
.SetDeviceOrigin( pt
.x
-x
, pt
.y 
); 
3847     wxArrayInt cols 
= m_owner
->CalcColLabelsExposed( GetUpdateRegion() ); 
3848     m_owner
->DrawColLabels( dc
, cols 
); 
3851 void wxGridColLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3853     m_owner
->ProcessColLabelMouseEvent( event 
); 
3856 void wxGridColLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3858     m_owner
->GetEventHandler()->ProcessEvent( event 
); 
3861 // This seems to be required for wxMotif otherwise the mouse 
3862 // cursor must be in the cell edit control to get key events 
3864 void wxGridColLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3866     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3870 void wxGridColLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3872     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3876 void wxGridColLabelWindow::OnChar( wxKeyEvent
& event 
) 
3878     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3882 ////////////////////////////////////////////////////////////////////// 
3884 IMPLEMENT_DYNAMIC_CLASS( wxGridCornerLabelWindow
, wxWindow 
) 
3886 BEGIN_EVENT_TABLE( wxGridCornerLabelWindow
, wxWindow 
) 
3887     EVT_MOUSEWHEEL( wxGridCornerLabelWindow::OnMouseWheel 
) 
3888     EVT_MOUSE_EVENTS( wxGridCornerLabelWindow::OnMouseEvent 
) 
3889     EVT_PAINT( wxGridCornerLabelWindow::OnPaint 
) 
3890     EVT_KEY_DOWN( wxGridCornerLabelWindow::OnKeyDown 
) 
3891     EVT_KEY_UP( wxGridCornerLabelWindow::OnKeyUp 
) 
3892     EVT_CHAR( wxGridCornerLabelWindow::OnChar 
) 
3895 wxGridCornerLabelWindow::wxGridCornerLabelWindow( wxGrid 
*parent
, 
3897                                                   const wxPoint 
&pos
, const wxSize 
&size 
) 
3898   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
| wxBORDER_NONE 
| wxFULL_REPAINT_ON_RESIZE 
) 
3903 void wxGridCornerLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3907     int client_height 
= 0; 
3908     int client_width 
= 0; 
3909     GetClientSize( &client_width
, &client_height 
); 
3911     // VZ: any reason for this ifdef? (FIXME) 
3917     rect
.SetWidth( client_width 
- 2 ); 
3918     rect
.SetHeight( client_height 
- 2 ); 
3920     wxRendererNative::Get().DrawHeaderButton( this, dc
, rect
, 0 ); 
3922     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DSHADOW
), 1, wxSOLID
) ); 
3923     dc
.DrawLine( client_width 
- 1, client_height 
- 1, client_width 
- 1, 0 ); 
3924     dc
.DrawLine( client_width 
- 1, client_height 
- 1, 0, client_height 
- 1 ); 
3925     dc
.DrawLine( 0, 0, client_width
, 0 ); 
3926     dc
.DrawLine( 0, 0, 0, client_height 
); 
3928     dc
.SetPen( *wxWHITE_PEN 
); 
3929     dc
.DrawLine( 1, 1, client_width 
- 1, 1 ); 
3930     dc
.DrawLine( 1, 1, 1, client_height 
- 1 ); 
3934 void wxGridCornerLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3936     m_owner
->ProcessCornerLabelMouseEvent( event 
); 
3939 void wxGridCornerLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3941     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3944 // This seems to be required for wxMotif otherwise the mouse 
3945 // cursor must be in the cell edit control to get key events 
3947 void wxGridCornerLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3949     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3953 void wxGridCornerLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3955     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3959 void wxGridCornerLabelWindow::OnChar( wxKeyEvent
& event 
) 
3961     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3965 ////////////////////////////////////////////////////////////////////// 
3967 IMPLEMENT_DYNAMIC_CLASS( wxGridWindow
, wxWindow 
) 
3969 BEGIN_EVENT_TABLE( wxGridWindow
, wxWindow 
) 
3970     EVT_PAINT( wxGridWindow::OnPaint 
) 
3971     EVT_MOUSEWHEEL( wxGridWindow::OnMouseWheel 
) 
3972     EVT_MOUSE_EVENTS( wxGridWindow::OnMouseEvent 
) 
3973     EVT_KEY_DOWN( wxGridWindow::OnKeyDown 
) 
3974     EVT_KEY_UP( wxGridWindow::OnKeyUp 
) 
3975     EVT_CHAR( wxGridWindow::OnChar 
) 
3976     EVT_SET_FOCUS( wxGridWindow::OnFocus 
) 
3977     EVT_KILL_FOCUS( wxGridWindow::OnFocus 
) 
3978     EVT_ERASE_BACKGROUND( wxGridWindow::OnEraseBackground 
) 
3981 wxGridWindow::wxGridWindow( wxGrid 
*parent
, 
3982                             wxGridRowLabelWindow 
*rowLblWin
, 
3983                             wxGridColLabelWindow 
*colLblWin
, 
3986                             const wxSize 
&size 
) 
3988                 parent
, id
, pos
, size
, 
3989                 wxWANTS_CHARS 
| wxBORDER_NONE 
| wxCLIP_CHILDREN 
| wxFULL_REPAINT_ON_RESIZE
, 
3990                 wxT("grid window") ) 
3993     m_rowLabelWin 
= rowLblWin
; 
3994     m_colLabelWin 
= colLblWin
; 
3997 void wxGridWindow::OnPaint( wxPaintEvent 
&WXUNUSED(event
) ) 
3999     wxPaintDC 
dc( this ); 
4000     m_owner
->PrepareDC( dc 
); 
4001     wxRegion reg 
= GetUpdateRegion(); 
4002     wxGridCellCoordsArray dirtyCells 
= m_owner
->CalcCellsExposed( reg 
); 
4003     m_owner
->DrawGridCellArea( dc
, dirtyCells 
); 
4005 #if WXGRID_DRAW_LINES 
4006     m_owner
->DrawAllGridLines( dc
, reg 
); 
4009     m_owner
->DrawGridSpace( dc 
); 
4010     m_owner
->DrawHighlight( dc
, dirtyCells 
); 
4013 void wxGridWindow::ScrollWindow( int dx
, int dy
, const wxRect 
*rect 
) 
4015     wxWindow::ScrollWindow( dx
, dy
, rect 
); 
4016     m_rowLabelWin
->ScrollWindow( 0, dy
, rect 
); 
4017     m_colLabelWin
->ScrollWindow( dx
, 0, rect 
); 
4020 void wxGridWindow::OnMouseEvent( wxMouseEvent
& event 
) 
4022     if (event
.ButtonDown(wxMOUSE_BTN_LEFT
) && FindFocus() != this) 
4025     m_owner
->ProcessGridCellMouseEvent( event 
); 
4028 void wxGridWindow::OnMouseWheel( wxMouseEvent
& event 
) 
4030     m_owner
->GetEventHandler()->ProcessEvent( event 
); 
4033 // This seems to be required for wxMotif/wxGTK otherwise the mouse 
4034 // cursor must be in the cell edit control to get key events 
4036 void wxGridWindow::OnKeyDown( wxKeyEvent
& event 
) 
4038     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
4042 void wxGridWindow::OnKeyUp( wxKeyEvent
& event 
) 
4044     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
4048 void wxGridWindow::OnChar( wxKeyEvent
& event 
) 
4050     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
4054 void wxGridWindow::OnEraseBackground( wxEraseEvent
& WXUNUSED(event
) ) 
4058 void wxGridWindow::OnFocus(wxFocusEvent
& event
) 
4060     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
4064 ////////////////////////////////////////////////////////////////////// 
4066 // Internal Helper function for computing row or column from some 
4067 // (unscrolled) coordinate value, using either 
4068 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
4069 // of m_rowBottoms/m_ColRights to speed up the search! 
4071 // Internal helper macros for simpler use of that function 
4073 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
4074                            const wxArrayInt
& BorderArray
, int nMax
, 
4077 #define internalXToCol(x) XToCol(x, true) 
4078 #define internalYToRow(y) CoordToRowOrCol(y, m_defaultRowHeight, \ 
4079                                           m_minAcceptableRowHeight, \ 
4080                                           m_rowBottoms, m_numRows, true) 
4082 ///////////////////////////////////////////////////////////////////// 
4084 #if wxUSE_EXTENDED_RTTI 
4085 WX_DEFINE_FLAGS( wxGridStyle 
) 
4087 wxBEGIN_FLAGS( wxGridStyle 
) 
4088     // new style border flags, we put them first to 
4089     // use them for streaming out 
4090     wxFLAGS_MEMBER(wxBORDER_SIMPLE
) 
4091     wxFLAGS_MEMBER(wxBORDER_SUNKEN
) 
4092     wxFLAGS_MEMBER(wxBORDER_DOUBLE
) 
4093     wxFLAGS_MEMBER(wxBORDER_RAISED
) 
4094     wxFLAGS_MEMBER(wxBORDER_STATIC
) 
4095     wxFLAGS_MEMBER(wxBORDER_NONE
) 
4097     // old style border flags 
4098     wxFLAGS_MEMBER(wxSIMPLE_BORDER
) 
4099     wxFLAGS_MEMBER(wxSUNKEN_BORDER
) 
4100     wxFLAGS_MEMBER(wxDOUBLE_BORDER
) 
4101     wxFLAGS_MEMBER(wxRAISED_BORDER
) 
4102     wxFLAGS_MEMBER(wxSTATIC_BORDER
) 
4103     wxFLAGS_MEMBER(wxBORDER
) 
4105     // standard window styles 
4106     wxFLAGS_MEMBER(wxTAB_TRAVERSAL
) 
4107     wxFLAGS_MEMBER(wxCLIP_CHILDREN
) 
4108     wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW
) 
4109     wxFLAGS_MEMBER(wxWANTS_CHARS
) 
4110     wxFLAGS_MEMBER(wxFULL_REPAINT_ON_RESIZE
) 
4111     wxFLAGS_MEMBER(wxALWAYS_SHOW_SB
) 
4112     wxFLAGS_MEMBER(wxVSCROLL
) 
4113     wxFLAGS_MEMBER(wxHSCROLL
) 
4115 wxEND_FLAGS( wxGridStyle 
) 
4117 IMPLEMENT_DYNAMIC_CLASS_XTI(wxGrid
, wxScrolledWindow
,"wx/grid.h") 
4119 wxBEGIN_PROPERTIES_TABLE(wxGrid
) 
4120     wxHIDE_PROPERTY( Children 
) 
4121     wxPROPERTY_FLAGS( WindowStyle 
, wxGridStyle 
, long , SetWindowStyleFlag 
, GetWindowStyleFlag 
, EMPTY_MACROVALUE
, 0 /*flags*/ , wxT("Helpstring") , wxT("group")) // style 
4122 wxEND_PROPERTIES_TABLE() 
4124 wxBEGIN_HANDLERS_TABLE(wxGrid
) 
4125 wxEND_HANDLERS_TABLE() 
4127 wxCONSTRUCTOR_5( wxGrid 
, wxWindow
* , Parent 
, wxWindowID 
, Id 
, wxPoint 
, Position 
, wxSize 
, Size 
, long , WindowStyle 
) 
4130  TODO : Expose more information of a list's layout, etc. via appropriate objects (e.g., NotebookPageInfo) 
4133 IMPLEMENT_DYNAMIC_CLASS( wxGrid
, wxScrolledWindow 
) 
4136 BEGIN_EVENT_TABLE( wxGrid
, wxScrolledWindow 
) 
4137     EVT_PAINT( wxGrid::OnPaint 
) 
4138     EVT_SIZE( wxGrid::OnSize 
) 
4139     EVT_KEY_DOWN( wxGrid::OnKeyDown 
) 
4140     EVT_KEY_UP( wxGrid::OnKeyUp 
) 
4141     EVT_CHAR ( wxGrid::OnChar 
) 
4142     EVT_ERASE_BACKGROUND( wxGrid::OnEraseBackground 
) 
4147     // in order to make sure that a size event is not 
4148     // trigerred in a unfinished state 
4149     m_cornerLabelWin 
= NULL
; 
4150     m_rowLabelWin 
= NULL
; 
4151     m_colLabelWin 
= NULL
; 
4155 wxGrid::wxGrid( wxWindow 
*parent
, 
4160                  const wxString
& name 
) 
4161   : wxScrolledWindow( parent
, id
, pos
, size
, (style 
| wxWANTS_CHARS
), name 
), 
4162     m_colMinWidths(GRID_HASH_SIZE
), 
4163     m_rowMinHeights(GRID_HASH_SIZE
) 
4166     SetInitialSize(size
); 
4169 bool wxGrid::Create(wxWindow 
*parent
, wxWindowID id
, 
4170                           const wxPoint
& pos
, const wxSize
& size
, 
4171                           long style
, const wxString
& name
) 
4173     if (!wxScrolledWindow::Create(parent
, id
, pos
, size
, 
4174                                   style 
| wxWANTS_CHARS
, name
)) 
4177     m_colMinWidths 
= wxLongToLongHashMap(GRID_HASH_SIZE
); 
4178     m_rowMinHeights 
= wxLongToLongHashMap(GRID_HASH_SIZE
); 
4181     SetInitialSize(size
); 
4188     // Must do this or ~wxScrollHelper will pop the wrong event handler 
4189     SetTargetWindow(this); 
4191     wxSafeDecRef(m_defaultCellAttr
); 
4193 #ifdef DEBUG_ATTR_CACHE 
4194     size_t total 
= gs_nAttrCacheHits 
+ gs_nAttrCacheMisses
; 
4195     wxPrintf(_T("wxGrid attribute cache statistics: " 
4196                 "total: %u, hits: %u (%u%%)\n"), 
4197              total
, gs_nAttrCacheHits
, 
4198              total 
? (gs_nAttrCacheHits
*100) / total 
: 0); 
4201     // if we own the table, just delete it, otherwise at least don't leave it 
4202     // with dangling view pointer 
4205     else if ( m_table 
&& m_table
->GetView() == this ) 
4206         m_table
->SetView(NULL
); 
4208     delete m_typeRegistry
; 
4213 // ----- internal init and update functions 
4216 // NOTE: If using the default visual attributes works everywhere then this can 
4217 // be removed as well as the #else cases below. 
4218 #define _USE_VISATTR 0 
4220 void wxGrid::Create() 
4222     // set to true by CreateGrid 
4225     // create the type registry 
4226     m_typeRegistry 
= new wxGridTypeRegistry
; 
4229     m_table 
= (wxGridTableBase 
*) NULL
; 
4232     m_cellEditCtrlEnabled 
= false; 
4234     m_defaultCellAttr 
= new wxGridCellAttr(); 
4236     // Set default cell attributes 
4237     m_defaultCellAttr
->SetDefAttr(m_defaultCellAttr
); 
4238     m_defaultCellAttr
->SetKind(wxGridCellAttr::Default
); 
4239     m_defaultCellAttr
->SetFont(GetFont()); 
4240     m_defaultCellAttr
->SetAlignment(wxALIGN_LEFT
, wxALIGN_TOP
); 
4241     m_defaultCellAttr
->SetRenderer(new wxGridCellStringRenderer
); 
4242     m_defaultCellAttr
->SetEditor(new wxGridCellTextEditor
); 
4245     wxVisualAttributes gva 
= wxListBox::GetClassDefaultAttributes(); 
4246     wxVisualAttributes lva 
= wxPanel::GetClassDefaultAttributes(); 
4248     m_defaultCellAttr
->SetTextColour(gva
.colFg
); 
4249     m_defaultCellAttr
->SetBackgroundColour(gva
.colBg
); 
4252     m_defaultCellAttr
->SetTextColour( 
4253         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT
)); 
4254     m_defaultCellAttr
->SetBackgroundColour( 
4255         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
4260     m_currentCellCoords 
= wxGridNoCellCoords
; 
4262     m_rowLabelWidth 
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
4263     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
4265     // subwindow components that make up the wxGrid 
4266     m_rowLabelWin 
= new wxGridRowLabelWindow( this, 
4271     m_colLabelWin 
= new wxGridColLabelWindow( this, 
4276     m_cornerLabelWin 
= new wxGridCornerLabelWindow( this, 
4281     m_gridWin 
= new wxGridWindow( this, 
4288     SetTargetWindow( m_gridWin 
); 
4291     wxColour gfg 
= gva
.colFg
; 
4292     wxColour gbg 
= gva
.colBg
; 
4293     wxColour lfg 
= lva
.colFg
; 
4294     wxColour lbg 
= lva
.colBg
; 
4296     wxColour gfg 
= wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOWTEXT 
); 
4297     wxColour gbg 
= wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW 
); 
4298     wxColour lfg 
= wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOWTEXT 
); 
4299     wxColour lbg 
= wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE 
); 
4302     m_cornerLabelWin
->SetOwnForegroundColour(lfg
); 
4303     m_cornerLabelWin
->SetOwnBackgroundColour(lbg
); 
4304     m_rowLabelWin
->SetOwnForegroundColour(lfg
); 
4305     m_rowLabelWin
->SetOwnBackgroundColour(lbg
); 
4306     m_colLabelWin
->SetOwnForegroundColour(lfg
); 
4307     m_colLabelWin
->SetOwnBackgroundColour(lbg
); 
4309     m_gridWin
->SetOwnForegroundColour(gfg
); 
4310     m_gridWin
->SetOwnBackgroundColour(gbg
); 
4315 bool wxGrid::CreateGrid( int numRows
, int numCols
, 
4316                          wxGrid::wxGridSelectionModes selmode 
) 
4318     wxCHECK_MSG( !m_created
, 
4320                  wxT("wxGrid::CreateGrid or wxGrid::SetTable called more than once") ); 
4322     m_numRows 
= numRows
; 
4323     m_numCols 
= numCols
; 
4325     m_table 
= new wxGridStringTable( m_numRows
, m_numCols 
); 
4326     m_table
->SetView( this ); 
4328     m_selection 
= new wxGridSelection( this, selmode 
); 
4337 void wxGrid::SetSelectionMode(wxGrid::wxGridSelectionModes selmode
) 
4339     wxCHECK_RET( m_created
, 
4340                  wxT("Called wxGrid::SetSelectionMode() before calling CreateGrid()") ); 
4342     m_selection
->SetSelectionMode( selmode 
); 
4345 wxGrid::wxGridSelectionModes 
wxGrid::GetSelectionMode() const 
4347     wxCHECK_MSG( m_created
, wxGrid::wxGridSelectCells
, 
4348                  wxT("Called wxGrid::GetSelectionMode() before calling CreateGrid()") ); 
4350     return m_selection
->GetSelectionMode(); 
4353 bool wxGrid::SetTable( wxGridTableBase 
*table
, bool takeOwnership
, 
4354                        wxGrid::wxGridSelectionModes selmode 
) 
4358         // stop all processing 
4363             wxGridTableBase 
*t 
= m_table
; 
4378         m_numRows 
= table
->GetNumberRows(); 
4379         m_numCols 
= table
->GetNumberCols(); 
4382         m_table
->SetView( this ); 
4383         m_ownTable 
= takeOwnership
; 
4384         m_selection 
= new wxGridSelection( this, selmode 
); 
4396     m_rowLabelWidth  
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
4397     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
4399     if ( m_rowLabelWin 
) 
4401         m_labelBackgroundColour 
= m_rowLabelWin
->GetBackgroundColour(); 
4405         m_labelBackgroundColour 
= *wxWHITE
; 
4408     m_labelTextColour 
= *wxBLACK
; 
4411     m_attrCache
.row 
= -1; 
4412     m_attrCache
.col 
= -1; 
4413     m_attrCache
.attr 
= NULL
; 
4415     // TODO: something better than this ? 
4417     m_labelFont 
= this->GetFont(); 
4418     m_labelFont
.SetWeight( wxBOLD 
); 
4420     m_rowLabelHorizAlign 
= wxALIGN_CENTRE
; 
4421     m_rowLabelVertAlign  
= wxALIGN_CENTRE
; 
4423     m_colLabelHorizAlign 
= wxALIGN_CENTRE
; 
4424     m_colLabelVertAlign  
= wxALIGN_CENTRE
; 
4425     m_colLabelTextOrientation 
= wxHORIZONTAL
; 
4427     m_defaultColWidth  
= WXGRID_DEFAULT_COL_WIDTH
; 
4428     m_defaultRowHeight 
= m_gridWin
->GetCharHeight(); 
4430     m_minAcceptableColWidth  
= WXGRID_MIN_COL_WIDTH
; 
4431     m_minAcceptableRowHeight 
= WXGRID_MIN_ROW_HEIGHT
; 
4433 #if defined(__WXMOTIF__) || defined(__WXGTK__)  // see also text ctrl sizing in ShowCellEditControl() 
4434     m_defaultRowHeight 
+= 8; 
4436     m_defaultRowHeight 
+= 4; 
4439     m_gridLineColour 
= wxColour( 192,192,192 ); 
4440     m_gridLinesEnabled 
= true; 
4441     m_cellHighlightColour 
= *wxBLACK
; 
4442     m_cellHighlightPenWidth 
= 2; 
4443     m_cellHighlightROPenWidth 
= 1; 
4445     m_canDragColMove 
= false; 
4447     m_cursorMode  
= WXGRID_CURSOR_SELECT_CELL
; 
4448     m_winCapture 
= (wxWindow 
*)NULL
; 
4449     m_canDragRowSize 
= true; 
4450     m_canDragColSize 
= true; 
4451     m_canDragGridSize 
= true; 
4452     m_canDragCell 
= false; 
4454     m_dragRowOrCol 
= -1; 
4455     m_isDragging 
= false; 
4456     m_startDragPos 
= wxDefaultPosition
; 
4458     m_waitForSlowClick 
= false; 
4460     m_rowResizeCursor 
= wxCursor( wxCURSOR_SIZENS 
); 
4461     m_colResizeCursor 
= wxCursor( wxCURSOR_SIZEWE 
); 
4463     m_currentCellCoords 
= wxGridNoCellCoords
; 
4467     m_selectionBackground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT
); 
4468     m_selectionForeground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
); 
4470     m_editable 
= true;  // default for whole grid 
4472     m_inOnKeyDown 
= false; 
4478     m_scrollLineX 
= GRID_SCROLL_LINE_X
; 
4479     m_scrollLineY 
= GRID_SCROLL_LINE_Y
; 
4482 // ---------------------------------------------------------------------------- 
4483 // the idea is to call these functions only when necessary because they create 
4484 // quite big arrays which eat memory mostly unnecessary - in particular, if 
4485 // default widths/heights are used for all rows/columns, we may not use these 
4488 // with some extra code, it should be possible to only store the 
4489 // widths/heights different from default ones but this will be done later... 
4490 // ---------------------------------------------------------------------------- 
4492 void wxGrid::InitRowHeights() 
4494     m_rowHeights
.Empty(); 
4495     m_rowBottoms
.Empty(); 
4497     m_rowHeights
.Alloc( m_numRows 
); 
4498     m_rowBottoms
.Alloc( m_numRows 
); 
4502     m_rowHeights
.Add( m_defaultRowHeight
, m_numRows 
); 
4504     for ( int i 
= 0; i 
< m_numRows
; i
++ ) 
4506         rowBottom 
+= m_defaultRowHeight
; 
4507         m_rowBottoms
.Add( rowBottom 
); 
4511 void wxGrid::InitColWidths() 
4513     m_colWidths
.Empty(); 
4514     m_colRights
.Empty(); 
4516     m_colWidths
.Alloc( m_numCols 
); 
4517     m_colRights
.Alloc( m_numCols 
); 
4520     m_colWidths
.Add( m_defaultColWidth
, m_numCols 
); 
4522     for ( int i 
= 0; i 
< m_numCols
; i
++ ) 
4524         colRight 
= ( GetColPos( i 
) + 1 ) * m_defaultColWidth
; 
4525         m_colRights
.Add( colRight 
); 
4529 int wxGrid::GetColWidth(int col
) const 
4531     return m_colWidths
.IsEmpty() ? m_defaultColWidth 
: m_colWidths
[col
]; 
4534 int wxGrid::GetColLeft(int col
) const 
4536     return m_colRights
.IsEmpty() ? GetColPos( col 
) * m_defaultColWidth
 
4537                                  : m_colRights
[col
] - m_colWidths
[col
]; 
4540 int wxGrid::GetColRight(int col
) const 
4542     return m_colRights
.IsEmpty() ? (GetColPos( col 
) + 1) * m_defaultColWidth
 
4546 int wxGrid::GetRowHeight(int row
) const 
4548     return m_rowHeights
.IsEmpty() ? m_defaultRowHeight 
: m_rowHeights
[row
]; 
4551 int wxGrid::GetRowTop(int row
) const 
4553     return m_rowBottoms
.IsEmpty() ? row 
* m_defaultRowHeight
 
4554                                   : m_rowBottoms
[row
] - m_rowHeights
[row
]; 
4557 int wxGrid::GetRowBottom(int row
) const 
4559     return m_rowBottoms
.IsEmpty() ? (row 
+ 1) * m_defaultRowHeight
 
4560                                   : m_rowBottoms
[row
]; 
4563 void wxGrid::CalcDimensions() 
4566     GetClientSize( &cw
, &ch 
); 
4568     if ( m_rowLabelWin
->IsShown() ) 
4569         cw 
-= m_rowLabelWidth
; 
4570     if ( m_colLabelWin
->IsShown() ) 
4571         ch 
-= m_colLabelHeight
; 
4574     int w 
= m_numCols 
> 0 ? GetColRight(GetColAt( m_numCols 
- 1 )) + m_extraWidth 
+ 1 : 0; 
4575     int h 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) + m_extraHeight 
+ 1 : 0; 
4577     // take into account editor if shown 
4578     if ( IsCellEditControlShown() ) 
4581         int r 
= m_currentCellCoords
.GetRow(); 
4582         int c 
= m_currentCellCoords
.GetCol(); 
4583         int x 
= GetColLeft(c
); 
4584         int y 
= GetRowTop(r
); 
4586         // how big is the editor 
4587         wxGridCellAttr
* attr 
= GetCellAttr(r
, c
); 
4588         wxGridCellEditor
* editor 
= attr
->GetEditor(this, r
, c
); 
4589         editor
->GetControl()->GetSize(&w2
, &h2
); 
4600     // preserve (more or less) the previous position 
4602     GetViewStart( &x
, &y 
); 
4604     // ensure the position is valid for the new scroll ranges 
4606         x 
= wxMax( w 
- 1, 0 ); 
4608         y 
= wxMax( h 
- 1, 0 ); 
4610     // do set scrollbar parameters 
4611     SetScrollbars( m_scrollLineX
, m_scrollLineY
, 
4612                    GetScrollX(w
), GetScrollY(h
), x
, y
, 
4613                    GetBatchCount() != 0); 
4615     // if our OnSize() hadn't been called (it would if we have scrollbars), we 
4616     // still must reposition the children 
4620 void wxGrid::CalcWindowSizes() 
4622     // escape if the window is has not been fully created yet 
4624     if ( m_cornerLabelWin 
== NULL 
) 
4628     GetClientSize( &cw
, &ch 
); 
4630     if ( m_cornerLabelWin 
&& m_cornerLabelWin
->IsShown() ) 
4631         m_cornerLabelWin
->SetSize( 0, 0, m_rowLabelWidth
, m_colLabelHeight 
); 
4633     if ( m_colLabelWin 
&& m_colLabelWin
->IsShown() ) 
4634         m_colLabelWin
->SetSize( m_rowLabelWidth
, 0, cw 
- m_rowLabelWidth
, m_colLabelHeight 
); 
4636     if ( m_rowLabelWin 
&& m_rowLabelWin
->IsShown() ) 
4637         m_rowLabelWin
->SetSize( 0, m_colLabelHeight
, m_rowLabelWidth
, ch 
- m_colLabelHeight 
); 
4639     if ( m_gridWin 
&& m_gridWin
->IsShown() ) 
4640         m_gridWin
->SetSize( m_rowLabelWidth
, m_colLabelHeight
, cw 
- m_rowLabelWidth
, ch 
- m_colLabelHeight 
); 
4643 // this is called when the grid table sends a message 
4644 // to indicate that it has been redimensioned 
4646 bool wxGrid::Redimension( wxGridTableMessage
& msg 
) 
4649     bool result 
= false; 
4651     // Clear the attribute cache as the attribute might refer to a different 
4652     // cell than stored in the cache after adding/removing rows/columns. 
4655     // By the same reasoning, the editor should be dismissed if columns are 
4656     // added or removed. And for consistency, it should IMHO always be 
4657     // removed, not only if the cell "underneath" it actually changes. 
4658     // For now, I intentionally do not save the editor's content as the 
4659     // cell it might want to save that stuff to might no longer exist. 
4660     HideCellEditControl(); 
4663     // if we were using the default widths/heights so far, we must change them 
4665     if ( m_colWidths
.IsEmpty() ) 
4670     if ( m_rowHeights
.IsEmpty() ) 
4676     switch ( msg
.GetId() ) 
4678         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
4680             size_t pos 
= msg
.GetCommandInt(); 
4681             int numRows 
= msg
.GetCommandInt2(); 
4683             m_numRows 
+= numRows
; 
4685             if ( !m_rowHeights
.IsEmpty() ) 
4687                 m_rowHeights
.Insert( m_defaultRowHeight
, pos
, numRows 
); 
4688                 m_rowBottoms
.Insert( 0, pos
, numRows 
); 
4692                     bottom 
= m_rowBottoms
[pos 
- 1]; 
4694                 for ( i 
= pos
; i 
< m_numRows
; i
++ ) 
4696                     bottom 
+= m_rowHeights
[i
]; 
4697                     m_rowBottoms
[i
] = bottom
; 
4701             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4703                 // if we have just inserted cols into an empty grid the current 
4704                 // cell will be undefined... 
4706                 SetCurrentCell( 0, 0 ); 
4710                 m_selection
->UpdateRows( pos
, numRows 
); 
4711             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4713                 attrProvider
->UpdateAttrRows( pos
, numRows 
); 
4715             if ( !GetBatchCount() ) 
4718                 m_rowLabelWin
->Refresh(); 
4724         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
4726             int numRows 
= msg
.GetCommandInt(); 
4727             int oldNumRows 
= m_numRows
; 
4728             m_numRows 
+= numRows
; 
4730             if ( !m_rowHeights
.IsEmpty() ) 
4732                 m_rowHeights
.Add( m_defaultRowHeight
, numRows 
); 
4733                 m_rowBottoms
.Add( 0, numRows 
); 
4736                 if ( oldNumRows 
> 0 ) 
4737                     bottom 
= m_rowBottoms
[oldNumRows 
- 1]; 
4739                 for ( i 
= oldNumRows
; i 
< m_numRows
; i
++ ) 
4741                     bottom 
+= m_rowHeights
[i
]; 
4742                     m_rowBottoms
[i
] = bottom
; 
4746             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4748                 // if we have just inserted cols into an empty grid the current 
4749                 // cell will be undefined... 
4751                 SetCurrentCell( 0, 0 ); 
4754             if ( !GetBatchCount() ) 
4757                 m_rowLabelWin
->Refresh(); 
4763         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
4765             size_t pos 
= msg
.GetCommandInt(); 
4766             int numRows 
= msg
.GetCommandInt2(); 
4767             m_numRows 
-= numRows
; 
4769             if ( !m_rowHeights
.IsEmpty() ) 
4771                 m_rowHeights
.RemoveAt( pos
, numRows 
); 
4772                 m_rowBottoms
.RemoveAt( pos
, numRows 
); 
4775                 for ( i 
= 0; i 
< m_numRows
; i
++ ) 
4777                     h 
+= m_rowHeights
[i
]; 
4778                     m_rowBottoms
[i
] = h
; 
4784                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4788                 if ( m_currentCellCoords
.GetRow() >= m_numRows 
) 
4789                     m_currentCellCoords
.Set( 0, 0 ); 
4793                 m_selection
->UpdateRows( pos
, -((int)numRows
) ); 
4794             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4797                 attrProvider
->UpdateAttrRows( pos
, -((int)numRows
) ); 
4799 // ifdef'd out following patch from Paul Gammans 
4801                 // No need to touch column attributes, unless we 
4802                 // removed _all_ rows, in this case, we remove 
4803                 // all column attributes. 
4804                 // I hate to do this here, but the 
4805                 // needed data is not available inside UpdateAttrRows. 
4806                 if ( !GetNumberRows() ) 
4807                     attrProvider
->UpdateAttrCols( 0, -GetNumberCols() ); 
4811             if ( !GetBatchCount() ) 
4814                 m_rowLabelWin
->Refresh(); 
4820         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
4822             size_t pos 
= msg
.GetCommandInt(); 
4823             int numCols 
= msg
.GetCommandInt2(); 
4824             m_numCols 
+= numCols
; 
4826             if ( !m_colAt
.IsEmpty() ) 
4828                 //Shift the column IDs 
4830                 for ( i 
= 0; i 
< m_numCols 
- numCols
; i
++ ) 
4832                     if ( m_colAt
[i
] >= (int)pos 
) 
4833                         m_colAt
[i
] += numCols
; 
4836                 m_colAt
.Insert( pos
, pos
, numCols 
); 
4838                 //Set the new columns' positions 
4839                 for ( i 
= pos 
+ 1; i 
< (int)pos 
+ numCols
; i
++ ) 
4845             if ( !m_colWidths
.IsEmpty() ) 
4847                 m_colWidths
.Insert( m_defaultColWidth
, pos
, numCols 
); 
4848                 m_colRights
.Insert( 0, pos
, numCols 
); 
4852                     right 
= m_colRights
[GetColAt( pos 
- 1 )]; 
4855                 for ( colPos 
= pos
; colPos 
< m_numCols
; colPos
++ ) 
4857                     i 
= GetColAt( colPos 
); 
4859                     right 
+= m_colWidths
[i
]; 
4860                     m_colRights
[i
] = right
; 
4864             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4866                 // if we have just inserted cols into an empty grid the current 
4867                 // cell will be undefined... 
4869                 SetCurrentCell( 0, 0 ); 
4873                 m_selection
->UpdateCols( pos
, numCols 
); 
4874             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4876                 attrProvider
->UpdateAttrCols( pos
, numCols 
); 
4877             if ( !GetBatchCount() ) 
4880                 m_colLabelWin
->Refresh(); 
4886         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
4888             int numCols 
= msg
.GetCommandInt(); 
4889             int oldNumCols 
= m_numCols
; 
4890             m_numCols 
+= numCols
; 
4892             if ( !m_colAt
.IsEmpty() ) 
4894                 m_colAt
.Add( 0, numCols 
); 
4896                 //Set the new columns' positions 
4898                 for ( i 
= oldNumCols
; i 
< m_numCols
; i
++ ) 
4904             if ( !m_colWidths
.IsEmpty() ) 
4906                 m_colWidths
.Add( m_defaultColWidth
, numCols 
); 
4907                 m_colRights
.Add( 0, numCols 
); 
4910                 if ( oldNumCols 
> 0 ) 
4911                     right 
= m_colRights
[GetColAt( oldNumCols 
- 1 )]; 
4914                 for ( colPos 
= oldNumCols
; colPos 
< m_numCols
; colPos
++ ) 
4916                     i 
= GetColAt( colPos 
); 
4918                     right 
+= m_colWidths
[i
]; 
4919                     m_colRights
[i
] = right
; 
4923             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4925                 // if we have just inserted cols into an empty grid the current 
4926                 // cell will be undefined... 
4928                 SetCurrentCell( 0, 0 ); 
4930             if ( !GetBatchCount() ) 
4933                 m_colLabelWin
->Refresh(); 
4939         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
4941             size_t pos 
= msg
.GetCommandInt(); 
4942             int numCols 
= msg
.GetCommandInt2(); 
4943             m_numCols 
-= numCols
; 
4945             if ( !m_colAt
.IsEmpty() ) 
4947                 int colID 
= GetColAt( pos 
); 
4949                 m_colAt
.RemoveAt( pos
, numCols 
); 
4951                 //Shift the column IDs 
4953                 for ( colPos 
= 0; colPos 
< m_numCols
; colPos
++ ) 
4955                     if ( m_colAt
[colPos
] > colID 
) 
4956                         m_colAt
[colPos
] -= numCols
; 
4960             if ( !m_colWidths
.IsEmpty() ) 
4962                 m_colWidths
.RemoveAt( pos
, numCols 
); 
4963                 m_colRights
.RemoveAt( pos
, numCols 
); 
4967                 for ( colPos 
= 0; colPos 
< m_numCols
; colPos
++ ) 
4969                     i 
= GetColAt( colPos 
); 
4971                     w 
+= m_colWidths
[i
]; 
4978                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4982                 if ( m_currentCellCoords
.GetCol() >= m_numCols 
) 
4983                   m_currentCellCoords
.Set( 0, 0 ); 
4987                 m_selection
->UpdateCols( pos
, -((int)numCols
) ); 
4988             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4991                 attrProvider
->UpdateAttrCols( pos
, -((int)numCols
) ); 
4993 // ifdef'd out following patch from Paul Gammans 
4995                 // No need to touch row attributes, unless we 
4996                 // removed _all_ columns, in this case, we remove 
4997                 // all row attributes. 
4998                 // I hate to do this here, but the 
4999                 // needed data is not available inside UpdateAttrCols. 
5000                 if ( !GetNumberCols() ) 
5001                     attrProvider
->UpdateAttrRows( 0, -GetNumberRows() ); 
5005             if ( !GetBatchCount() ) 
5008                 m_colLabelWin
->Refresh(); 
5015     if (result 
&& !GetBatchCount() ) 
5016         m_gridWin
->Refresh(); 
5021 wxArrayInt 
wxGrid::CalcRowLabelsExposed( const wxRegion
& reg 
) 
5023     wxRegionIterator 
iter( reg 
); 
5026     wxArrayInt  rowlabels
; 
5033         // TODO: remove this when we can... 
5034         // There is a bug in wxMotif that gives garbage update 
5035         // rectangles if you jump-scroll a long way by clicking the 
5036         // scrollbar with middle button.  This is a work-around 
5038 #if defined(__WXMOTIF__) 
5040         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5041         if ( r
.GetTop() > ch 
) 
5043         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
5046         // logical bounds of update region 
5049         CalcUnscrolledPosition( 0, r
.GetTop(), &dummy
, &top 
); 
5050         CalcUnscrolledPosition( 0, r
.GetBottom(), &dummy
, &bottom 
); 
5052         // find the row labels within these bounds 
5055         for ( row 
= internalYToRow(top
); row 
< m_numRows
; row
++ ) 
5057             if ( GetRowBottom(row
) < top 
) 
5060             if ( GetRowTop(row
) > bottom 
) 
5063             rowlabels
.Add( row 
); 
5072 wxArrayInt 
wxGrid::CalcColLabelsExposed( const wxRegion
& reg 
) 
5074     wxRegionIterator 
iter( reg 
); 
5077     wxArrayInt colLabels
; 
5084         // TODO: remove this when we can... 
5085         // There is a bug in wxMotif that gives garbage update 
5086         // rectangles if you jump-scroll a long way by clicking the 
5087         // scrollbar with middle button.  This is a work-around 
5089 #if defined(__WXMOTIF__) 
5091         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5092         if ( r
.GetLeft() > cw 
) 
5094         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
5097         // logical bounds of update region 
5100         CalcUnscrolledPosition( r
.GetLeft(), 0, &left
, &dummy 
); 
5101         CalcUnscrolledPosition( r
.GetRight(), 0, &right
, &dummy 
); 
5103         // find the cells within these bounds 
5107         for ( colPos 
= GetColPos( internalXToCol(left
) ); colPos 
< m_numCols
; colPos
++ ) 
5109             col 
= GetColAt( colPos 
); 
5111             if ( GetColRight(col
) < left 
) 
5114             if ( GetColLeft(col
) > right 
) 
5117             colLabels
.Add( col 
); 
5126 wxGridCellCoordsArray 
wxGrid::CalcCellsExposed( const wxRegion
& reg 
) 
5128     wxRegionIterator 
iter( reg 
); 
5131     wxGridCellCoordsArray  cellsExposed
; 
5133     int left
, top
, right
, bottom
; 
5138         // TODO: remove this when we can... 
5139         // There is a bug in wxMotif that gives garbage update 
5140         // rectangles if you jump-scroll a long way by clicking the 
5141         // scrollbar with middle button.  This is a work-around 
5143 #if defined(__WXMOTIF__) 
5145         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5146         if ( r
.GetTop() > ch 
) r
.SetTop( 0 ); 
5147         if ( r
.GetLeft() > cw 
) r
.SetLeft( 0 ); 
5148         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
5149         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
5152         // logical bounds of update region 
5154         CalcUnscrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
5155         CalcUnscrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
5157         // find the cells within these bounds 
5160         for ( row 
= internalYToRow(top
); row 
< m_numRows
; row
++ ) 
5162             if ( GetRowBottom(row
) <= top 
) 
5165             if ( GetRowTop(row
) > bottom 
) 
5169             for ( colPos 
= GetColPos( internalXToCol(left
) ); colPos 
< m_numCols
; colPos
++ ) 
5171                 col 
= GetColAt( colPos 
); 
5173                 if ( GetColRight(col
) <= left 
) 
5176                 if ( GetColLeft(col
) > right 
) 
5179                 cellsExposed
.Add( wxGridCellCoords( row
, col 
) ); 
5186     return cellsExposed
; 
5190 void wxGrid::ProcessRowLabelMouseEvent( wxMouseEvent
& event 
) 
5193     wxPoint 
pos( event
.GetPosition() ); 
5194     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
5196     if ( event
.Dragging() ) 
5200             m_isDragging 
= true; 
5201             m_rowLabelWin
->CaptureMouse(); 
5204         if ( event
.LeftIsDown() ) 
5206             switch ( m_cursorMode 
) 
5208                 case WXGRID_CURSOR_RESIZE_ROW
: 
5210                     int cw
, ch
, left
, dummy
; 
5211                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
5212                     CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5214                     wxClientDC 
dc( m_gridWin 
); 
5217                                GetRowTop(m_dragRowOrCol
) + 
5218                                GetRowMinimalHeight(m_dragRowOrCol
) ); 
5219                     dc
.SetLogicalFunction(wxINVERT
); 
5220                     if ( m_dragLastPos 
>= 0 ) 
5222                         dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5224                     dc
.DrawLine( left
, y
, left
+cw
, y 
); 
5229                 case WXGRID_CURSOR_SELECT_ROW
: 
5231                     if ( (row 
= YToRow( y 
)) >= 0 ) 
5235                             m_selection
->SelectRow( row
, 
5236                                                     event
.ControlDown(), 
5245                 // default label to suppress warnings about "enumeration value 
5246                 // 'xxx' not handled in switch 
5254     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
5259         if (m_rowLabelWin
->HasCapture()) 
5260             m_rowLabelWin
->ReleaseMouse(); 
5261         m_isDragging 
= false; 
5264     // ------------ Entering or leaving the window 
5266     if ( event
.Entering() || event
.Leaving() ) 
5268         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
5271     // ------------ Left button pressed 
5273     else if ( event
.LeftDown() ) 
5275         // don't send a label click event for a hit on the 
5276         // edge of the row label - this is probably the user 
5277         // wanting to resize the row 
5279         if ( YToEdgeOfRow(y
) < 0 ) 
5283                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, row
, -1, event 
) ) 
5285                 if ( !event
.ShiftDown() && !event
.CmdDown() ) 
5289                     if ( event
.ShiftDown() ) 
5291                         m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
5294                                                   GetNumberCols() - 1, 
5295                                                   event
.ControlDown(), 
5302                         m_selection
->SelectRow( row
, 
5303                                                 event
.ControlDown(), 
5310                 ChangeCursorMode(WXGRID_CURSOR_SELECT_ROW
, m_rowLabelWin
); 
5315             // starting to drag-resize a row 
5316             if ( CanDragRowSize() ) 
5317                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
); 
5321     // ------------ Left double click 
5323     else if (event
.LeftDClick() ) 
5325         row 
= YToEdgeOfRow(y
); 
5330                  !SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, row
, -1, event 
) ) 
5332                 // no default action at the moment 
5337             // adjust row height depending on label text 
5338             AutoSizeRowLabelSize( row 
); 
5340             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5345     // ------------ Left button released 
5347     else if ( event
.LeftUp() ) 
5349         if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5351             DoEndDragResizeRow(); 
5353             // Note: we are ending the event *after* doing 
5354             // default processing in this case 
5356             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
5359         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
5363     // ------------ Right button down 
5365     else if ( event
.RightDown() ) 
5369              !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, row
, -1, event 
) ) 
5371             // no default action at the moment 
5375     // ------------ Right double click 
5377     else if ( event
.RightDClick() ) 
5381              !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, row
, -1, event 
) ) 
5383             // no default action at the moment 
5387     // ------------ No buttons down and mouse moving 
5389     else if ( event
.Moving() ) 
5391         m_dragRowOrCol 
= YToEdgeOfRow( y 
); 
5392         if ( m_dragRowOrCol 
>= 0 ) 
5394             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5396                 // don't capture the mouse yet 
5397                 if ( CanDragRowSize() ) 
5398                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
, false); 
5401         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
5403             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
, false); 
5408 void wxGrid::ProcessColLabelMouseEvent( wxMouseEvent
& event 
) 
5411     wxPoint 
pos( event
.GetPosition() ); 
5412     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
5414     if ( event
.Dragging() ) 
5418             m_isDragging 
= true; 
5419             m_colLabelWin
->CaptureMouse(); 
5421             if ( m_cursorMode 
== WXGRID_CURSOR_MOVE_COL 
) 
5422                 m_dragRowOrCol 
= XToCol( x 
); 
5425         if ( event
.LeftIsDown() ) 
5427             switch ( m_cursorMode 
) 
5429                 case WXGRID_CURSOR_RESIZE_COL
: 
5431                     int cw
, ch
, dummy
, top
; 
5432                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
5433                     CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5435                     wxClientDC 
dc( m_gridWin 
); 
5438                     x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
5439                                   GetColMinimalWidth(m_dragRowOrCol
)); 
5440                     dc
.SetLogicalFunction(wxINVERT
); 
5441                     if ( m_dragLastPos 
>= 0 ) 
5443                         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top 
+ ch 
); 
5445                     dc
.DrawLine( x
, top
, x
, top 
+ ch 
); 
5450                 case WXGRID_CURSOR_SELECT_COL
: 
5452                     if ( (col 
= XToCol( x 
)) >= 0 ) 
5456                             m_selection
->SelectCol( col
, 
5457                                                     event
.ControlDown(), 
5466                 case WXGRID_CURSOR_MOVE_COL
: 
5469                         m_moveToCol 
= GetColAt( 0 ); 
5471                         m_moveToCol 
= XToCol( x 
); 
5475                     if ( m_moveToCol 
< 0 ) 
5476                         markerX 
= GetColRight( GetColAt( m_numCols 
- 1 ) ); 
5478                         markerX 
= GetColLeft( m_moveToCol 
); 
5480                     if ( markerX 
!= m_dragLastPos 
) 
5482                         wxClientDC 
dc( m_colLabelWin 
); 
5485                         m_colLabelWin
->GetClientSize( &cw
, &ch 
); 
5489                         //Clean up the last indicator 
5490                         if ( m_dragLastPos 
>= 0 ) 
5492                             wxPen 
pen( m_colLabelWin
->GetBackgroundColour(), 2 ); 
5494                             dc
.DrawLine( m_dragLastPos 
+ 1, 0, m_dragLastPos 
+ 1, ch 
); 
5495                             dc
.SetPen(wxNullPen
); 
5497                             if ( XToCol( m_dragLastPos 
) != -1 ) 
5498                                 DrawColLabel( dc
, XToCol( m_dragLastPos 
) ); 
5501                         //Moving to the same place? Don't draw a marker 
5502                         if ( (m_moveToCol 
== m_dragRowOrCol
) 
5503                           || (GetColPos( m_moveToCol 
) == GetColPos( m_dragRowOrCol 
) + 1) 
5504                           || (m_moveToCol 
< 0 && m_dragRowOrCol 
== GetColAt( m_numCols 
- 1 ))) 
5511                         wxPen 
pen( *wxBLUE
, 2 ); 
5514                         dc
.DrawLine( markerX
, 0, markerX
, ch 
); 
5516                         dc
.SetPen(wxNullPen
); 
5518                         m_dragLastPos 
= markerX 
- 1; 
5523                 // default label to suppress warnings about "enumeration value 
5524                 // 'xxx' not handled in switch 
5532     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
5537         if (m_colLabelWin
->HasCapture()) 
5538             m_colLabelWin
->ReleaseMouse(); 
5539         m_isDragging 
= false; 
5542     // ------------ Entering or leaving the window 
5544     if ( event
.Entering() || event
.Leaving() ) 
5546         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5549     // ------------ Left button pressed 
5551     else if ( event
.LeftDown() ) 
5553         // don't send a label click event for a hit on the 
5554         // edge of the col label - this is probably the user 
5555         // wanting to resize the col 
5557         if ( XToEdgeOfCol(x
) < 0 ) 
5561                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, col
, event 
) ) 
5563                 if ( m_canDragColMove 
) 
5565                     //Show button as pressed 
5566                     wxClientDC 
dc( m_colLabelWin 
); 
5567                     int colLeft 
= GetColLeft( col 
); 
5568                     int colRight 
= GetColRight( col 
) - 1; 
5569                     dc
.SetPen( wxPen( m_colLabelWin
->GetBackgroundColour(), 1 ) ); 
5570                     dc
.DrawLine( colLeft
, 1, colLeft
, m_colLabelHeight
-1 ); 
5571                     dc
.DrawLine( colLeft
, 1, colRight
, 1 ); 
5573                     ChangeCursorMode(WXGRID_CURSOR_MOVE_COL
, m_colLabelWin
); 
5577                     if ( !event
.ShiftDown() && !event
.CmdDown() ) 
5581                         if ( event
.ShiftDown() ) 
5583                             m_selection
->SelectBlock( 0, 
5584                                                       m_currentCellCoords
.GetCol(), 
5585                                                       GetNumberRows() - 1, col
, 
5586                                                       event
.ControlDown(), 
5593                             m_selection
->SelectCol( col
, 
5594                                                     event
.ControlDown(), 
5601                     ChangeCursorMode(WXGRID_CURSOR_SELECT_COL
, m_colLabelWin
); 
5607             // starting to drag-resize a col 
5609             if ( CanDragColSize() ) 
5610                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
); 
5614     // ------------ Left double click 
5616     if ( event
.LeftDClick() ) 
5618         col 
= XToEdgeOfCol(x
); 
5623                  ! SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, col
, event 
) ) 
5625                 // no default action at the moment 
5630             // adjust column width depending on label text 
5631             AutoSizeColLabelSize( col 
); 
5633             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5638     // ------------ Left button released 
5640     else if ( event
.LeftUp() ) 
5642         switch ( m_cursorMode 
) 
5644             case WXGRID_CURSOR_RESIZE_COL
: 
5645                 DoEndDragResizeCol(); 
5647                 // Note: we are ending the event *after* doing 
5648                 // default processing in this case 
5650                 SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
5653             case WXGRID_CURSOR_MOVE_COL
: 
5656                 SendEvent( wxEVT_GRID_COL_MOVE
, -1, m_dragRowOrCol
, event 
); 
5659             case WXGRID_CURSOR_SELECT_COL
: 
5660             case WXGRID_CURSOR_SELECT_CELL
: 
5661             case WXGRID_CURSOR_RESIZE_ROW
: 
5662             case WXGRID_CURSOR_SELECT_ROW
: 
5663                 // nothing to do (?) 
5667         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5671     // ------------ Right button down 
5673     else if ( event
.RightDown() ) 
5677              !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, col
, event 
) ) 
5679             // no default action at the moment 
5683     // ------------ Right double click 
5685     else if ( event
.RightDClick() ) 
5689              !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, col
, event 
) ) 
5691             // no default action at the moment 
5695     // ------------ No buttons down and mouse moving 
5697     else if ( event
.Moving() ) 
5699         m_dragRowOrCol 
= XToEdgeOfCol( x 
); 
5700         if ( m_dragRowOrCol 
>= 0 ) 
5702             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5704                 // don't capture the cursor yet 
5705                 if ( CanDragColSize() ) 
5706                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
, false); 
5709         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
5711             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
, false); 
5716 void wxGrid::ProcessCornerLabelMouseEvent( wxMouseEvent
& event 
) 
5718     if ( event
.LeftDown() ) 
5720         // indicate corner label by having both row and 
5723         if ( !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, -1, event 
) ) 
5728     else if ( event
.LeftDClick() ) 
5730         SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, -1, event 
); 
5732     else if ( event
.RightDown() ) 
5734         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, -1, event 
) ) 
5736             // no default action at the moment 
5739     else if ( event
.RightDClick() ) 
5741         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, -1, event 
) ) 
5743             // no default action at the moment 
5748 void wxGrid::ChangeCursorMode(CursorMode mode
, 
5753     static const wxChar 
*cursorModes
[] = 
5763     wxLogTrace(_T("grid"), 
5764                _T("wxGrid cursor mode (mouse capture for %s): %s -> %s"), 
5765                win 
== m_colLabelWin 
? _T("colLabelWin") 
5766                                     : win 
? _T("rowLabelWin") 
5768                cursorModes
[m_cursorMode
], cursorModes
[mode
]); 
5771     if ( mode 
== m_cursorMode 
&& 
5772          win 
== m_winCapture 
&& 
5773          captureMouse 
== (m_winCapture 
!= NULL
)) 
5778         // by default use the grid itself 
5784         if (m_winCapture
->HasCapture()) 
5785             m_winCapture
->ReleaseMouse(); 
5786         m_winCapture 
= (wxWindow 
*)NULL
; 
5789     m_cursorMode 
= mode
; 
5791     switch ( m_cursorMode 
) 
5793         case WXGRID_CURSOR_RESIZE_ROW
: 
5794             win
->SetCursor( m_rowResizeCursor 
); 
5797         case WXGRID_CURSOR_RESIZE_COL
: 
5798             win
->SetCursor( m_colResizeCursor 
); 
5801         case WXGRID_CURSOR_MOVE_COL
: 
5802             win
->SetCursor( wxCursor(wxCURSOR_HAND
) ); 
5806             win
->SetCursor( *wxSTANDARD_CURSOR 
); 
5810     // we need to capture mouse when resizing 
5811     bool resize 
= m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
|| 
5812                   m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL
; 
5814     if ( captureMouse 
&& resize 
) 
5816         win
->CaptureMouse(); 
5821 void wxGrid::ProcessGridCellMouseEvent( wxMouseEvent
& event 
) 
5824     wxPoint 
pos( event
.GetPosition() ); 
5825     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
5827     wxGridCellCoords coords
; 
5828     XYToCell( x
, y
, coords 
); 
5830     int cell_rows
, cell_cols
; 
5831     bool isFirstDrag 
= !m_isDragging
; 
5832     GetCellSize( coords
.GetRow(), coords
.GetCol(), &cell_rows
, &cell_cols 
); 
5833     if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
5835         coords
.SetRow(coords
.GetRow() + cell_rows
); 
5836         coords
.SetCol(coords
.GetCol() + cell_cols
); 
5839     if ( event
.Dragging() ) 
5841         //wxLogDebug("pos(%d, %d) coords(%d, %d)", pos.x, pos.y, coords.GetRow(), coords.GetCol()); 
5843         // Don't start doing anything until the mouse has been dragged at 
5844         // least 3 pixels in any direction... 
5847             if (m_startDragPos 
== wxDefaultPosition
) 
5849                 m_startDragPos 
= pos
; 
5852             if (abs(m_startDragPos
.x 
- pos
.x
) < 4 && abs(m_startDragPos
.y 
- pos
.y
) < 4) 
5856         m_isDragging 
= true; 
5857         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5859             // Hide the edit control, so it 
5860             // won't interfere with drag-shrinking. 
5861             if ( IsCellEditControlShown() ) 
5863                 HideCellEditControl(); 
5864                 SaveEditControlValue(); 
5867             // Have we captured the mouse yet? 
5870                 m_winCapture 
= m_gridWin
; 
5871                 m_winCapture
->CaptureMouse(); 
5874             if ( coords 
!= wxGridNoCellCoords 
) 
5876                 if ( event
.CmdDown() ) 
5878                     if ( m_selectingKeyboard 
== wxGridNoCellCoords
) 
5879                         m_selectingKeyboard 
= coords
; 
5880                     HighlightBlock( m_selectingKeyboard
, coords 
); 
5882                 else if ( CanDragCell() ) 
5886                         if ( m_selectingKeyboard 
== wxGridNoCellCoords
) 
5887                             m_selectingKeyboard 
= coords
; 
5889                         SendEvent( wxEVT_GRID_CELL_BEGIN_DRAG
, 
5897                     if ( !IsSelection() ) 
5899                         HighlightBlock( coords
, coords 
); 
5903                         HighlightBlock( m_currentCellCoords
, coords 
); 
5907                 if (! IsVisible(coords
)) 
5909                     MakeCellVisible(coords
); 
5910                     // TODO: need to introduce a delay or something here.  The 
5911                     // scrolling is way to fast, at least on MSW - also on GTK. 
5915         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5917             int cw
, ch
, left
, dummy
; 
5918             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5919             CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5921             wxClientDC 
dc( m_gridWin 
); 
5923             y 
= wxMax( y
, GetRowTop(m_dragRowOrCol
) + 
5924                           GetRowMinimalHeight(m_dragRowOrCol
) ); 
5925             dc
.SetLogicalFunction(wxINVERT
); 
5926             if ( m_dragLastPos 
>= 0 ) 
5928                 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5930             dc
.DrawLine( left
, y
, left
+cw
, y 
); 
5933         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5935             int cw
, ch
, dummy
, top
; 
5936             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5937             CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5939             wxClientDC 
dc( m_gridWin 
); 
5941             x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
5942                           GetColMinimalWidth(m_dragRowOrCol
) ); 
5943             dc
.SetLogicalFunction(wxINVERT
); 
5944             if ( m_dragLastPos 
>= 0 ) 
5946                 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top 
+ ch 
); 
5948             dc
.DrawLine( x
, top
, x
, top 
+ ch 
); 
5955     m_isDragging 
= false; 
5956     m_startDragPos 
= wxDefaultPosition
; 
5958     // VZ: if we do this, the mode is reset to WXGRID_CURSOR_SELECT_CELL 
5959     //     immediately after it becomes WXGRID_CURSOR_RESIZE_ROW/COL under 
5962     if ( event
.Entering() || event
.Leaving() ) 
5964         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5965         m_gridWin
->SetCursor( *wxSTANDARD_CURSOR 
); 
5970     // ------------ Left button pressed 
5972     if ( event
.LeftDown() && coords 
!= wxGridNoCellCoords 
) 
5974         if ( !SendEvent( wxEVT_GRID_CELL_LEFT_CLICK
, 
5979             if ( !event
.CmdDown() ) 
5981             if ( event
.ShiftDown() ) 
5985                     m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
5986                                               m_currentCellCoords
.GetCol(), 
5989                                               event
.ControlDown(), 
5995             else if ( XToEdgeOfCol(x
) < 0 && 
5996                       YToEdgeOfRow(y
) < 0 ) 
5998                 DisableCellEditControl(); 
5999                 MakeCellVisible( coords 
); 
6001                 if ( event
.CmdDown() ) 
6005                         m_selection
->ToggleCellSelection( coords
.GetRow(), 
6007                                                           event
.ControlDown(), 
6012                     m_selectingTopLeft 
= wxGridNoCellCoords
; 
6013                     m_selectingBottomRight 
= wxGridNoCellCoords
; 
6014                     m_selectingKeyboard 
= coords
; 
6018                     m_waitForSlowClick 
= m_currentCellCoords 
== coords 
&& coords 
!= wxGridNoCellCoords
; 
6019                     SetCurrentCell( coords 
); 
6022                         if ( m_selection
->GetSelectionMode() != 
6023                                 wxGrid::wxGridSelectCells 
) 
6025                             HighlightBlock( coords
, coords 
); 
6033     // ------------ Left double click 
6035     else if ( event
.LeftDClick() && coords 
!= wxGridNoCellCoords 
) 
6037         DisableCellEditControl(); 
6039         if ( XToEdgeOfCol(x
) < 0 && YToEdgeOfRow(y
) < 0 ) 
6041             if ( !SendEvent( wxEVT_GRID_CELL_LEFT_DCLICK
, 
6046                 // we want double click to select a cell and start editing 
6047                 // (i.e. to behave in same way as sequence of two slow clicks): 
6048                 m_waitForSlowClick 
= true; 
6053     // ------------ Left button released 
6055     else if ( event
.LeftUp() ) 
6057         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
6061                 if (m_winCapture
->HasCapture()) 
6062                     m_winCapture
->ReleaseMouse(); 
6063                 m_winCapture 
= NULL
; 
6066             if ( coords 
== m_currentCellCoords 
&& m_waitForSlowClick 
&& CanEnableCellControl() ) 
6069                 EnableCellEditControl(); 
6071                 wxGridCellAttr 
*attr 
= GetCellAttr(coords
); 
6072                 wxGridCellEditor 
*editor 
= attr
->GetEditor(this, coords
.GetRow(), coords
.GetCol()); 
6073                 editor
->StartingClick(); 
6077                 m_waitForSlowClick 
= false; 
6079             else if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
6080                  m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
6084                     m_selection
->SelectBlock( m_selectingTopLeft
.GetRow(), 
6085                                               m_selectingTopLeft
.GetCol(), 
6086                                               m_selectingBottomRight
.GetRow(), 
6087                                               m_selectingBottomRight
.GetCol(), 
6088                                               event
.ControlDown(), 
6094                 m_selectingTopLeft 
= wxGridNoCellCoords
; 
6095                 m_selectingBottomRight 
= wxGridNoCellCoords
; 
6097                 // Show the edit control, if it has been hidden for 
6099                 ShowCellEditControl(); 
6102         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
6104             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
6105             DoEndDragResizeRow(); 
6107             // Note: we are ending the event *after* doing 
6108             // default processing in this case 
6110             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
6112         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
6114             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
6115             DoEndDragResizeCol(); 
6117             // Note: we are ending the event *after* doing 
6118             // default processing in this case 
6120             SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
6126     // ------------ Right button down 
6128     else if ( event
.RightDown() && coords 
!= wxGridNoCellCoords 
) 
6130         DisableCellEditControl(); 
6131         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_CLICK
, 
6136             // no default action at the moment 
6140     // ------------ Right double click 
6142     else if ( event
.RightDClick() && coords 
!= wxGridNoCellCoords 
) 
6144         DisableCellEditControl(); 
6145         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_DCLICK
, 
6150             // no default action at the moment 
6154     // ------------ Moving and no button action 
6156     else if ( event
.Moving() && !event
.IsButton() ) 
6158         if ( coords
.GetRow() < 0 || coords
.GetCol() < 0 ) 
6160             // out of grid cell area 
6161             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
6165         int dragRow 
= YToEdgeOfRow( y 
); 
6166         int dragCol 
= XToEdgeOfCol( x 
); 
6168         // Dragging on the corner of a cell to resize in both 
6169         // directions is not implemented yet... 
6171         if ( dragRow 
>= 0 && dragCol 
>= 0 ) 
6173             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
6179             m_dragRowOrCol 
= dragRow
; 
6181             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
6183                 if ( CanDragRowSize() && CanDragGridSize() ) 
6184                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
); 
6187         else if ( dragCol 
>= 0 ) 
6189             m_dragRowOrCol 
= dragCol
; 
6191             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
6193                 if ( CanDragColSize() && CanDragGridSize() ) 
6194                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
); 
6197         else // Neither on a row or col edge 
6199             if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
6201                 ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
6207 void wxGrid::DoEndDragResizeRow() 
6209     if ( m_dragLastPos 
>= 0 ) 
6211         // erase the last line and resize the row 
6213         int cw
, ch
, left
, dummy
; 
6214         m_gridWin
->GetClientSize( &cw
, &ch 
); 
6215         CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
6217         wxClientDC 
dc( m_gridWin 
); 
6219         dc
.SetLogicalFunction( wxINVERT 
); 
6220         dc
.DrawLine( left
, m_dragLastPos
, left 
+ cw
, m_dragLastPos 
); 
6221         HideCellEditControl(); 
6222         SaveEditControlValue(); 
6224         int rowTop 
= GetRowTop(m_dragRowOrCol
); 
6225         SetRowSize( m_dragRowOrCol
, 
6226                     wxMax( m_dragLastPos 
- rowTop
, m_minAcceptableRowHeight 
) ); 
6228         if ( !GetBatchCount() ) 
6230             // Only needed to get the correct rect.y: 
6231             wxRect 
rect ( CellToRect( m_dragRowOrCol
, 0 ) ); 
6233             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
6234             rect
.width 
= m_rowLabelWidth
; 
6235             rect
.height 
= ch 
- rect
.y
; 
6236             m_rowLabelWin
->Refresh( true, &rect 
); 
6239             // if there is a multicell block, paint all of it 
6242                 int i
, cell_rows
, cell_cols
, subtract_rows 
= 0; 
6243                 int leftCol 
= XToCol(left
); 
6244                 int rightCol 
= internalXToCol(left 
+ cw
); 
6247                     for (i
=leftCol
; i
<rightCol
; i
++) 
6249                         GetCellSize(m_dragRowOrCol
, i
, &cell_rows
, &cell_cols
); 
6250                         if (cell_rows 
< subtract_rows
) 
6251                             subtract_rows 
= cell_rows
; 
6253                     rect
.y 
= GetRowTop(m_dragRowOrCol 
+ subtract_rows
); 
6254                     CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
6255                     rect
.height 
= ch 
- rect
.y
; 
6258             m_gridWin
->Refresh( false, &rect 
); 
6261         ShowCellEditControl(); 
6266 void wxGrid::DoEndDragResizeCol() 
6268     if ( m_dragLastPos 
>= 0 ) 
6270         // erase the last line and resize the col 
6272         int cw
, ch
, dummy
, top
; 
6273         m_gridWin
->GetClientSize( &cw
, &ch 
); 
6274         CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
6276         wxClientDC 
dc( m_gridWin 
); 
6278         dc
.SetLogicalFunction( wxINVERT 
); 
6279         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top 
+ ch 
); 
6280         HideCellEditControl(); 
6281         SaveEditControlValue(); 
6283         int colLeft 
= GetColLeft(m_dragRowOrCol
); 
6284         SetColSize( m_dragRowOrCol
, 
6285                     wxMax( m_dragLastPos 
- colLeft
, 
6286                            GetColMinimalWidth(m_dragRowOrCol
) ) ); 
6288         if ( !GetBatchCount() ) 
6290             // Only needed to get the correct rect.x: 
6291             wxRect 
rect ( CellToRect( 0, m_dragRowOrCol 
) ); 
6293             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
6294             rect
.width 
= cw 
- rect
.x
; 
6295             rect
.height 
= m_colLabelHeight
; 
6296             m_colLabelWin
->Refresh( true, &rect 
); 
6299             // if there is a multicell block, paint all of it 
6302                 int i
, cell_rows
, cell_cols
, subtract_cols 
= 0; 
6303                 int topRow 
= YToRow(top
); 
6304                 int bottomRow 
= internalYToRow(top 
+ cw
); 
6307                     for (i
=topRow
; i
<bottomRow
; i
++) 
6309                         GetCellSize(i
, m_dragRowOrCol
, &cell_rows
, &cell_cols
); 
6310                         if (cell_cols 
< subtract_cols
) 
6311                             subtract_cols 
= cell_cols
; 
6314                     rect
.x 
= GetColLeft(m_dragRowOrCol 
+ subtract_cols
); 
6315                     CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
6316                     rect
.width 
= cw 
- rect
.x
; 
6320             m_gridWin
->Refresh( false, &rect 
); 
6323         ShowCellEditControl(); 
6327 void wxGrid::DoEndDragMoveCol() 
6329     //The user clicked on the column but didn't actually drag 
6330     if ( m_dragLastPos 
< 0 ) 
6332         m_colLabelWin
->Refresh();   //Do this to "unpress" the column 
6337     if ( m_moveToCol 
== -1 ) 
6338         newPos 
= m_numCols 
- 1; 
6341         newPos 
= GetColPos( m_moveToCol 
); 
6342         if ( newPos 
> GetColPos( m_dragRowOrCol 
) ) 
6346     SetColPos( m_dragRowOrCol
, newPos 
); 
6349 void wxGrid::SetColPos( int colID
, int newPos 
) 
6351     if ( m_colAt
.IsEmpty() ) 
6353         m_colAt
.Alloc( m_numCols 
); 
6356         for ( i 
= 0; i 
< m_numCols
; i
++ ) 
6362     int oldPos 
= GetColPos( colID 
); 
6364     //Reshuffle the m_colAt array 
6365     if ( newPos 
> oldPos 
) 
6368         for ( i 
= oldPos
; i 
< newPos
; i
++ ) 
6370             m_colAt
[i
] = m_colAt
[i
+1]; 
6376         for ( i 
= oldPos
; i 
> newPos
; i
-- ) 
6378             m_colAt
[i
] = m_colAt
[i
-1]; 
6382     m_colAt
[newPos
] = colID
; 
6384     //Recalculate the column rights 
6385     if ( !m_colWidths
.IsEmpty() ) 
6389         for ( colPos 
= 0; colPos 
< m_numCols
; colPos
++ ) 
6391             int colID 
= GetColAt( colPos 
); 
6393             colRight 
+= m_colWidths
[colID
]; 
6394             m_colRights
[colID
] = colRight
; 
6398     m_colLabelWin
->Refresh(); 
6399     m_gridWin
->Refresh(); 
6404 void wxGrid::EnableDragColMove( bool enable 
) 
6406     if ( m_canDragColMove 
== enable 
) 
6409     m_canDragColMove 
= enable
; 
6411     if ( !m_canDragColMove 
) 
6415         //Recalculate the column rights 
6416         if ( !m_colWidths
.IsEmpty() ) 
6420             for ( colPos 
= 0; colPos 
< m_numCols
; colPos
++ ) 
6422                 colRight 
+= m_colWidths
[colPos
]; 
6423                 m_colRights
[colPos
] = colRight
; 
6427         m_colLabelWin
->Refresh(); 
6428         m_gridWin
->Refresh(); 
6434 // ------ interaction with data model 
6436 bool wxGrid::ProcessTableMessage( wxGridTableMessage
& msg 
) 
6438     switch ( msg
.GetId() ) 
6440         case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES
: 
6441             return GetModelValues(); 
6443         case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES
: 
6444             return SetModelValues(); 
6446         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
6447         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
6448         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
6449         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
6450         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
6451         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
6452             return Redimension( msg 
); 
6459 // The behaviour of this function depends on the grid table class 
6460 // Clear() function. For the default wxGridStringTable class the 
6461 // behavious is to replace all cell contents with wxEmptyString but 
6462 // not to change the number of rows or cols. 
6464 void wxGrid::ClearGrid() 
6468         if (IsCellEditControlEnabled()) 
6469             DisableCellEditControl(); 
6472         if (!GetBatchCount()) 
6473             m_gridWin
->Refresh(); 
6477 bool wxGrid::InsertRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
6479     // TODO: something with updateLabels flag 
6483         wxFAIL_MSG( wxT("Called wxGrid::InsertRows() before calling CreateGrid()") ); 
6489         if (IsCellEditControlEnabled()) 
6490             DisableCellEditControl(); 
6492         bool done 
= m_table
->InsertRows( pos
, numRows 
); 
6495         // the table will have sent the results of the insert row 
6496         // operation to this view object as a grid table message 
6502 bool wxGrid::AppendRows( int numRows
, bool WXUNUSED(updateLabels
) ) 
6504     // TODO: something with updateLabels flag 
6508         wxFAIL_MSG( wxT("Called wxGrid::AppendRows() before calling CreateGrid()") ); 
6514         bool done 
= m_table 
&& m_table
->AppendRows( numRows 
); 
6517         // the table will have sent the results of the append row 
6518         // operation to this view object as a grid table message 
6524 bool wxGrid::DeleteRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
6526     // TODO: something with updateLabels flag 
6530         wxFAIL_MSG( wxT("Called wxGrid::DeleteRows() before calling CreateGrid()") ); 
6536         if (IsCellEditControlEnabled()) 
6537             DisableCellEditControl(); 
6539         bool done 
= m_table
->DeleteRows( pos
, numRows 
); 
6541         // the table will have sent the results of the delete row 
6542         // operation to this view object as a grid table message 
6548 bool wxGrid::InsertCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
6550     // TODO: something with updateLabels flag 
6554         wxFAIL_MSG( wxT("Called wxGrid::InsertCols() before calling CreateGrid()") ); 
6560         if (IsCellEditControlEnabled()) 
6561             DisableCellEditControl(); 
6563         bool done 
= m_table
->InsertCols( pos
, numCols 
); 
6565         // the table will have sent the results of the insert col 
6566         // operation to this view object as a grid table message 
6572 bool wxGrid::AppendCols( int numCols
, bool WXUNUSED(updateLabels
) ) 
6574     // TODO: something with updateLabels flag 
6578         wxFAIL_MSG( wxT("Called wxGrid::AppendCols() before calling CreateGrid()") ); 
6584         bool done 
= m_table
->AppendCols( numCols 
); 
6586         // the table will have sent the results of the append col 
6587         // operation to this view object as a grid table message 
6593 bool wxGrid::DeleteCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
6595     // TODO: something with updateLabels flag 
6599         wxFAIL_MSG( wxT("Called wxGrid::DeleteCols() before calling CreateGrid()") ); 
6605         if (IsCellEditControlEnabled()) 
6606             DisableCellEditControl(); 
6608         bool done 
= m_table
->DeleteCols( pos
, numCols 
); 
6610         // the table will have sent the results of the delete col 
6611         // operation to this view object as a grid table message 
6618 // ----- event handlers 
6621 // Generate a grid event based on a mouse event and 
6622 // return the result of ProcessEvent() 
6624 int wxGrid::SendEvent( const wxEventType type
, 
6626                         wxMouseEvent
& mouseEv 
) 
6628    bool claimed
, vetoed
; 
6630    if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
6632        int rowOrCol 
= (row 
== -1 ? col 
: row
); 
6634        wxGridSizeEvent 
gridEvt( GetId(), 
6638                mouseEv
.GetX() + GetRowLabelSize(), 
6639                mouseEv
.GetY() + GetColLabelSize(), 
6640                mouseEv
.ControlDown(), 
6641                mouseEv
.ShiftDown(), 
6643                mouseEv
.MetaDown() ); 
6645        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6646        vetoed 
= !gridEvt
.IsAllowed(); 
6648    else if ( type 
== wxEVT_GRID_RANGE_SELECT 
) 
6650        // Right now, it should _never_ end up here! 
6651        wxGridRangeSelectEvent 
gridEvt( GetId(), 
6655                m_selectingBottomRight
, 
6657                mouseEv
.ControlDown(), 
6658                mouseEv
.ShiftDown(), 
6660                mouseEv
.MetaDown() ); 
6662        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6663        vetoed 
= !gridEvt
.IsAllowed(); 
6665    else if ( type 
== wxEVT_GRID_LABEL_LEFT_CLICK 
|| 
6666              type 
== wxEVT_GRID_LABEL_LEFT_DCLICK 
|| 
6667              type 
== wxEVT_GRID_LABEL_RIGHT_CLICK 
|| 
6668              type 
== wxEVT_GRID_LABEL_RIGHT_DCLICK 
) 
6670        wxPoint pos 
= mouseEv
.GetPosition(); 
6672        if ( mouseEv
.GetEventObject() == GetGridRowLabelWindow() ) 
6673            pos
.y 
+= GetColLabelSize(); 
6674        if ( mouseEv
.GetEventObject() == GetGridColLabelWindow() ) 
6675            pos
.x 
+= GetRowLabelSize(); 
6677        wxGridEvent 
gridEvt( GetId(), 
6684                mouseEv
.ControlDown(), 
6685                mouseEv
.ShiftDown(), 
6687                mouseEv
.MetaDown() ); 
6688        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6689        vetoed 
= !gridEvt
.IsAllowed(); 
6693        wxGridEvent 
gridEvt( GetId(), 
6697                mouseEv
.GetX() + GetRowLabelSize(), 
6698                mouseEv
.GetY() + GetColLabelSize(), 
6700                mouseEv
.ControlDown(), 
6701                mouseEv
.ShiftDown(), 
6703                mouseEv
.MetaDown() ); 
6704        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6705        vetoed 
= !gridEvt
.IsAllowed(); 
6708    // A Veto'd event may not be `claimed' so test this first 
6712    return claimed 
? 1 : 0; 
6715 // Generate a grid event of specified type and return the result 
6716 // of ProcessEvent(). 
6718 int wxGrid::SendEvent( const wxEventType type
, 
6721    bool claimed
, vetoed
; 
6723     if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
6725         int rowOrCol 
= (row 
== -1 ? col 
: row
); 
6727         wxGridSizeEvent 
gridEvt( GetId(), type
, this, rowOrCol 
); 
6729         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6730         vetoed  
= !gridEvt
.IsAllowed(); 
6734         wxGridEvent 
gridEvt( GetId(), type
, this, row
, col 
); 
6736         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6737         vetoed  
= !gridEvt
.IsAllowed(); 
6740     // A Veto'd event may not be `claimed' so test this first 
6744     return claimed 
? 1 : 0; 
6747 void wxGrid::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
6749     // needed to prevent zillions of paint events on MSW 
6753 void wxGrid::Refresh(bool eraseb
, const wxRect
* rect
) 
6755     // Don't do anything if between Begin/EndBatch... 
6756     // EndBatch() will do all this on the last nested one anyway. 
6757     if (! GetBatchCount()) 
6759         // Refresh to get correct scrolled position: 
6760         wxScrolledWindow::Refresh(eraseb
, rect
); 
6764             int rect_x
, rect_y
, rectWidth
, rectHeight
; 
6765             int width_label
, width_cell
, height_label
, height_cell
; 
6768             // Copy rectangle can get scroll offsets.. 
6769             rect_x 
= rect
->GetX(); 
6770             rect_y 
= rect
->GetY(); 
6771             rectWidth 
= rect
->GetWidth(); 
6772             rectHeight 
= rect
->GetHeight(); 
6774             width_label 
= m_rowLabelWidth 
- rect_x
; 
6775             if (width_label 
> rectWidth
) 
6776                 width_label 
= rectWidth
; 
6778             height_label 
= m_colLabelHeight 
- rect_y
; 
6779             if (height_label 
> rectHeight
) 
6780                 height_label 
= rectHeight
; 
6782             if (rect_x 
> m_rowLabelWidth
) 
6784                 x 
= rect_x 
- m_rowLabelWidth
; 
6785                 width_cell 
= rectWidth
; 
6790                 width_cell 
= rectWidth 
- (m_rowLabelWidth 
- rect_x
); 
6793             if (rect_y 
> m_colLabelHeight
) 
6795                 y 
= rect_y 
- m_colLabelHeight
; 
6796                 height_cell 
= rectHeight
; 
6801                 height_cell 
= rectHeight 
- (m_colLabelHeight 
- rect_y
); 
6804             // Paint corner label part intersecting rect. 
6805             if ( width_label 
> 0 && height_label 
> 0 ) 
6807                 wxRect 
anotherrect(rect_x
, rect_y
, width_label
, height_label
); 
6808                 m_cornerLabelWin
->Refresh(eraseb
, &anotherrect
); 
6811             // Paint col labels part intersecting rect. 
6812             if ( width_cell 
> 0 && height_label 
> 0 ) 
6814                 wxRect 
anotherrect(x
, rect_y
, width_cell
, height_label
); 
6815                 m_colLabelWin
->Refresh(eraseb
, &anotherrect
); 
6818             // Paint row labels part intersecting rect. 
6819             if ( width_label 
> 0 && height_cell 
> 0 ) 
6821                 wxRect 
anotherrect(rect_x
, y
, width_label
, height_cell
); 
6822                 m_rowLabelWin
->Refresh(eraseb
, &anotherrect
); 
6825             // Paint cell area part intersecting rect. 
6826             if ( width_cell 
> 0 && height_cell 
> 0 ) 
6828                 wxRect 
anotherrect(x
, y
, width_cell
, height_cell
); 
6829                 m_gridWin
->Refresh(eraseb
, &anotherrect
); 
6834             m_cornerLabelWin
->Refresh(eraseb
, NULL
); 
6835             m_colLabelWin
->Refresh(eraseb
, NULL
); 
6836             m_rowLabelWin
->Refresh(eraseb
, NULL
); 
6837             m_gridWin
->Refresh(eraseb
, NULL
); 
6842 void wxGrid::OnSize( wxSizeEvent
& event 
) 
6844     // position the child windows 
6847     // don't call CalcDimensions() from here, the base class handles the size 
6852 void wxGrid::OnKeyDown( wxKeyEvent
& event 
) 
6854     if ( m_inOnKeyDown 
) 
6856         // shouldn't be here - we are going round in circles... 
6858         wxFAIL_MSG( wxT("wxGrid::OnKeyDown called while already active") ); 
6861     m_inOnKeyDown 
= true; 
6863     // propagate the event up and see if it gets processed 
6864     wxWindow 
*parent 
= GetParent(); 
6865     wxKeyEvent 
keyEvt( event 
); 
6866     keyEvt
.SetEventObject( parent 
); 
6868     if ( !parent
->GetEventHandler()->ProcessEvent( keyEvt 
) ) 
6870         if (GetLayoutDirection() == wxLayout_RightToLeft
) 
6872             if (event
.GetKeyCode() == WXK_RIGHT
) 
6873                 event
.m_keyCode 
= WXK_LEFT
; 
6874             else if (event
.GetKeyCode() == WXK_LEFT
) 
6875                 event
.m_keyCode 
= WXK_RIGHT
; 
6878         // try local handlers 
6879         switch ( event
.GetKeyCode() ) 
6882                 if ( event
.ControlDown() ) 
6883                     MoveCursorUpBlock( event
.ShiftDown() ); 
6885                     MoveCursorUp( event
.ShiftDown() ); 
6889                 if ( event
.ControlDown() ) 
6890                     MoveCursorDownBlock( event
.ShiftDown() ); 
6892                     MoveCursorDown( event
.ShiftDown() ); 
6896                 if ( event
.ControlDown() ) 
6897                     MoveCursorLeftBlock( event
.ShiftDown() ); 
6899                     MoveCursorLeft( event
.ShiftDown() ); 
6903                 if ( event
.ControlDown() ) 
6904                     MoveCursorRightBlock( event
.ShiftDown() ); 
6906                     MoveCursorRight( event
.ShiftDown() ); 
6910             case WXK_NUMPAD_ENTER
: 
6911                 if ( event
.ControlDown() ) 
6913                     event
.Skip();  // to let the edit control have the return 
6917                     if ( GetGridCursorRow() < GetNumberRows()-1 ) 
6919                         MoveCursorDown( event
.ShiftDown() ); 
6923                         // at the bottom of a column 
6924                         DisableCellEditControl(); 
6934                 if (event
.ShiftDown()) 
6936                     if ( GetGridCursorCol() > 0 ) 
6938                         MoveCursorLeft( false ); 
6943                         DisableCellEditControl(); 
6948                     if ( GetGridCursorCol() < GetNumberCols() - 1 ) 
6950                         MoveCursorRight( false ); 
6955                         DisableCellEditControl(); 
6961                 if ( event
.ControlDown() ) 
6963                     MakeCellVisible( 0, 0 ); 
6964                     SetCurrentCell( 0, 0 ); 
6973                 if ( event
.ControlDown() ) 
6975                     MakeCellVisible( m_numRows 
- 1, m_numCols 
- 1 ); 
6976                     SetCurrentCell( m_numRows 
- 1, m_numCols 
- 1 ); 
6993                 if ( event
.ControlDown() ) 
6997                         m_selection
->ToggleCellSelection( 
6998                             m_currentCellCoords
.GetRow(), 
6999                             m_currentCellCoords
.GetCol(), 
7000                             event
.ControlDown(), 
7008                 if ( !IsEditable() ) 
7009                     MoveCursorRight( false ); 
7020     m_inOnKeyDown 
= false; 
7023 void wxGrid::OnKeyUp( wxKeyEvent
& event 
) 
7025     // try local handlers 
7027     if ( event
.GetKeyCode() == WXK_SHIFT 
) 
7029         if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
7030              m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
7034                 m_selection
->SelectBlock( 
7035                     m_selectingTopLeft
.GetRow(), 
7036                     m_selectingTopLeft
.GetCol(), 
7037                     m_selectingBottomRight
.GetRow(), 
7038                     m_selectingBottomRight
.GetCol(), 
7039                     event
.ControlDown(), 
7046         m_selectingTopLeft 
= wxGridNoCellCoords
; 
7047         m_selectingBottomRight 
= wxGridNoCellCoords
; 
7048         m_selectingKeyboard 
= wxGridNoCellCoords
; 
7052 void wxGrid::OnChar( wxKeyEvent
& event 
) 
7054     // is it possible to edit the current cell at all? 
7055     if ( !IsCellEditControlEnabled() && CanEnableCellControl() ) 
7057         // yes, now check whether the cells editor accepts the key 
7058         int row 
= m_currentCellCoords
.GetRow(); 
7059         int col 
= m_currentCellCoords
.GetCol(); 
7060         wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
7061         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
7063         // <F2> is special and will always start editing, for 
7064         // other keys - ask the editor itself 
7065         if ( (event
.GetKeyCode() == WXK_F2 
&& !event
.HasModifiers()) 
7066              || editor
->IsAcceptedKey(event
) ) 
7068             // ensure cell is visble 
7069             MakeCellVisible(row
, col
); 
7070             EnableCellEditControl(); 
7072             // a problem can arise if the cell is not completely 
7073             // visible (even after calling MakeCellVisible the 
7074             // control is not created and calling StartingKey will 
7076             if ( event
.GetKeyCode() != WXK_F2 
&& editor
->IsCreated() && m_cellEditCtrlEnabled 
) 
7077                 editor
->StartingKey(event
); 
7093 void wxGrid::OnEraseBackground(wxEraseEvent
&) 
7097 void wxGrid::SetCurrentCell( const wxGridCellCoords
& coords 
) 
7099     if ( SendEvent( wxEVT_GRID_SELECT_CELL
, coords
.GetRow(), coords
.GetCol() ) ) 
7101         // the event has been intercepted - do nothing 
7105 #if !(defined(__WXMAC__) && wxMAC_USE_CORE_GRAPHICS) 
7106     wxClientDC 
dc( m_gridWin 
); 
7110     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
) 
7112         DisableCellEditControl(); 
7114         if ( IsVisible( m_currentCellCoords
, false ) ) 
7117             r 
= BlockToDeviceRect( m_currentCellCoords
, m_currentCellCoords 
); 
7118             if ( !m_gridLinesEnabled 
) 
7126             wxGridCellCoordsArray cells 
= CalcCellsExposed( r 
); 
7128             // Otherwise refresh redraws the highlight! 
7129             m_currentCellCoords 
= coords
; 
7131 #if defined(__WXMAC__) && wxMAC_USE_CORE_GRAPHICS 
7132             m_gridWin
->Refresh(true /*, & r */); 
7134             DrawGridCellArea( dc
, cells 
); 
7135             DrawAllGridLines( dc
, r 
); 
7140     m_currentCellCoords 
= coords
; 
7142     wxGridCellAttr 
*attr 
= GetCellAttr( coords 
); 
7143 #if !(defined(__WXMAC__) && wxMAC_USE_CORE_GRAPHICS) 
7144     DrawCellHighlight( dc
, attr 
); 
7149 void wxGrid::HighlightBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol 
) 
7152     wxGridCellCoords updateTopLeft
, updateBottomRight
; 
7156         if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
7159             rightCol 
= GetNumberCols() - 1; 
7161         else if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
7164             bottomRow 
= GetNumberRows() - 1; 
7168     if ( topRow 
> bottomRow 
) 
7175     if ( leftCol 
> rightCol 
) 
7182     updateTopLeft 
= wxGridCellCoords( topRow
, leftCol 
); 
7183     updateBottomRight 
= wxGridCellCoords( bottomRow
, rightCol 
); 
7185     // First the case that we selected a completely new area 
7186     if ( m_selectingTopLeft 
== wxGridNoCellCoords 
|| 
7187          m_selectingBottomRight 
== wxGridNoCellCoords 
) 
7190         rect 
= BlockToDeviceRect( wxGridCellCoords ( topRow
, leftCol 
), 
7191                                   wxGridCellCoords ( bottomRow
, rightCol 
) ); 
7192         m_gridWin
->Refresh( false, &rect 
); 
7195     // Now handle changing an existing selection area. 
7196     else if ( m_selectingTopLeft 
!= updateTopLeft 
|| 
7197               m_selectingBottomRight 
!= updateBottomRight 
) 
7199         // Compute two optimal update rectangles: 
7200         // Either one rectangle is a real subset of the 
7201         // other, or they are (almost) disjoint! 
7203         bool    need_refresh
[4]; 
7207         need_refresh
[3] = false; 
7210         // Store intermediate values 
7211         wxCoord oldLeft 
= m_selectingTopLeft
.GetCol(); 
7212         wxCoord oldTop 
= m_selectingTopLeft
.GetRow(); 
7213         wxCoord oldRight 
= m_selectingBottomRight
.GetCol(); 
7214         wxCoord oldBottom 
= m_selectingBottomRight
.GetRow(); 
7216         // Determine the outer/inner coordinates. 
7217         if (oldLeft 
> leftCol
) 
7223         if (oldTop 
> topRow 
) 
7229         if (oldRight 
< rightCol 
) 
7232             oldRight 
= rightCol
; 
7235         if (oldBottom 
< bottomRow
) 
7238             oldBottom 
= bottomRow
; 
7242         // Now, either the stuff marked old is the outer 
7243         // rectangle or we don't have a situation where one 
7244         // is contained in the other. 
7246         if ( oldLeft 
< leftCol 
) 
7248             // Refresh the newly selected or deselected 
7249             // area to the left of the old or new selection. 
7250             need_refresh
[0] = true; 
7251             rect
[0] = BlockToDeviceRect( 
7252                 wxGridCellCoords( oldTop
,  oldLeft 
), 
7253                 wxGridCellCoords( oldBottom
, leftCol 
- 1 ) ); 
7256         if ( oldTop 
< topRow 
) 
7258             // Refresh the newly selected or deselected 
7259             // area above the old or new selection. 
7260             need_refresh
[1] = true; 
7261             rect
[1] = BlockToDeviceRect( 
7262                 wxGridCellCoords( oldTop
, leftCol 
), 
7263                 wxGridCellCoords( topRow 
- 1, rightCol 
) ); 
7266         if ( oldRight 
> rightCol 
) 
7268             // Refresh the newly selected or deselected 
7269             // area to the right of the old or new selection. 
7270             need_refresh
[2] = true; 
7271             rect
[2] = BlockToDeviceRect( 
7272                 wxGridCellCoords( oldTop
, rightCol 
+ 1 ), 
7273                 wxGridCellCoords( oldBottom
, oldRight 
) ); 
7276         if ( oldBottom 
> bottomRow 
) 
7278             // Refresh the newly selected or deselected 
7279             // area below the old or new selection. 
7280             need_refresh
[3] = true; 
7281             rect
[3] = BlockToDeviceRect( 
7282                 wxGridCellCoords( bottomRow 
+ 1, leftCol 
), 
7283                 wxGridCellCoords( oldBottom
, rightCol 
) ); 
7286         // various Refresh() calls 
7287         for (i 
= 0; i 
< 4; i
++ ) 
7288             if ( need_refresh
[i
] && rect
[i
] != wxGridNoCellRect 
) 
7289                 m_gridWin
->Refresh( false, &(rect
[i
]) ); 
7293     m_selectingTopLeft 
= updateTopLeft
; 
7294     m_selectingBottomRight 
= updateBottomRight
; 
7298 // ------ functions to get/send data (see also public functions) 
7301 bool wxGrid::GetModelValues() 
7303     // Hide the editor, so it won't hide a changed value. 
7304     HideCellEditControl(); 
7308         // all we need to do is repaint the grid 
7310         m_gridWin
->Refresh(); 
7317 bool wxGrid::SetModelValues() 
7321     // Disable the editor, so it won't hide a changed value. 
7322     // Do we also want to save the current value of the editor first? 
7324     DisableCellEditControl(); 
7328         for ( row 
= 0; row 
< m_numRows
; row
++ ) 
7330             for ( col 
= 0; col 
< m_numCols
; col
++ ) 
7332                 m_table
->SetValue( row
, col
, GetCellValue(row
, col
) ); 
7342 // Note - this function only draws cells that are in the list of 
7343 // exposed cells (usually set from the update region by 
7344 // CalcExposedCells) 
7346 void wxGrid::DrawGridCellArea( wxDC
& dc
, const wxGridCellCoordsArray
& cells 
) 
7348     if ( !m_numRows 
|| !m_numCols 
) 
7351     int i
, numCells 
= cells
.GetCount(); 
7352     int row
, col
, cell_rows
, cell_cols
; 
7353     wxGridCellCoordsArray redrawCells
; 
7355     for ( i 
= numCells 
- 1; i 
>= 0; i
-- ) 
7357         row 
= cells
[i
].GetRow(); 
7358         col 
= cells
[i
].GetCol(); 
7359         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7361         // If this cell is part of a multicell block, find owner for repaint 
7362         if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
7364             wxGridCellCoords 
cell( row 
+ cell_rows
, col 
+ cell_cols 
); 
7365             bool marked 
= false; 
7366             for ( int j 
= 0; j 
< numCells
; j
++ ) 
7368                 if ( cell 
== cells
[j
] ) 
7377                 int count 
= redrawCells
.GetCount(); 
7378                 for (int j 
= 0; j 
< count
; j
++) 
7380                     if ( cell 
== redrawCells
[j
] ) 
7388                     redrawCells
.Add( cell 
); 
7391             // don't bother drawing this cell 
7395         // If this cell is empty, find cell to left that might want to overflow 
7396         if (m_table 
&& m_table
->IsEmptyCell(row
, col
)) 
7398             for ( int l 
= 0; l 
< cell_rows
; l
++ ) 
7400                 // find a cell in this row to leave already marked for repaint 
7402                 for (int k 
= 0; k 
< int(redrawCells
.GetCount()); k
++) 
7403                     if ((redrawCells
[k
].GetCol() < left
) && 
7404                         (redrawCells
[k
].GetRow() == row
)) 
7406                         left 
= redrawCells
[k
].GetCol(); 
7410                     left 
= 0; // oh well 
7412                 for (int j 
= col 
- 1; j 
>= left
; j
--) 
7414                     if (!m_table
->IsEmptyCell(row 
+ l
, j
)) 
7416                         if (GetCellOverflow(row 
+ l
, j
)) 
7418                             wxGridCellCoords 
cell(row 
+ l
, j
); 
7419                             bool marked 
= false; 
7421                             for (int k 
= 0; k 
< numCells
; k
++) 
7423                                 if ( cell 
== cells
[k
] ) 
7432                                 int count 
= redrawCells
.GetCount(); 
7433                                 for (int k 
= 0; k 
< count
; k
++) 
7435                                     if ( cell 
== redrawCells
[k
] ) 
7442                                     redrawCells
.Add( cell 
); 
7451         DrawCell( dc
, cells
[i
] ); 
7454     numCells 
= redrawCells
.GetCount(); 
7456     for ( i 
= numCells 
- 1; i 
>= 0; i
-- ) 
7458         DrawCell( dc
, redrawCells
[i
] ); 
7462 void wxGrid::DrawGridSpace( wxDC
& dc 
) 
7465   m_gridWin
->GetClientSize( &cw
, &ch 
); 
7468   CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
7470   int rightCol 
= m_numCols 
> 0 ? GetColRight(GetColAt( m_numCols 
- 1 )) : 0; 
7471   int bottomRow 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) : 0; 
7473   if ( right 
> rightCol 
|| bottom 
> bottomRow 
) 
7476       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
7478       dc
.SetBrush( wxBrush(GetDefaultCellBackgroundColour(), wxSOLID
) ); 
7479       dc
.SetPen( *wxTRANSPARENT_PEN 
); 
7481       if ( right 
> rightCol 
) 
7483           dc
.DrawRectangle( rightCol
, top
, right 
- rightCol
, ch 
); 
7486       if ( bottom 
> bottomRow 
) 
7488           dc
.DrawRectangle( left
, bottomRow
, cw
, bottom 
- bottomRow 
); 
7493 void wxGrid::DrawCell( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
7495     int row 
= coords
.GetRow(); 
7496     int col 
= coords
.GetCol(); 
7498     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
7501     // we draw the cell border ourselves 
7502 #if !WXGRID_DRAW_LINES 
7503     if ( m_gridLinesEnabled 
) 
7504         DrawCellBorder( dc
, coords 
); 
7507     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7509     bool isCurrent 
= coords 
== m_currentCellCoords
; 
7511     wxRect rect 
= CellToRect( row
, col 
); 
7513     // if the editor is shown, we should use it and not the renderer 
7514     // Note: However, only if it is really _shown_, i.e. not hidden! 
7515     if ( isCurrent 
&& IsCellEditControlShown() ) 
7517         // NB: this "#if..." is temporary and fixes a problem where the 
7518         // edit control is erased by this code after being rendered. 
7519         // On wxMac (QD build only), the cell editor is a wxTextCntl and is rendered 
7520         // implicitly, causing this out-of order render. 
7521 #if !defined(__WXMAC__) 
7522         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
7523         editor
->PaintBackground(rect
, attr
); 
7529         // but all the rest is drawn by the cell renderer and hence may be customized 
7530         wxGridCellRenderer 
*renderer 
= attr
->GetRenderer(this, row
, col
); 
7531         renderer
->Draw(*this, *attr
, dc
, rect
, row
, col
, IsInSelection(coords
)); 
7538 void wxGrid::DrawCellHighlight( wxDC
& dc
, const wxGridCellAttr 
*attr 
) 
7540     int row 
= m_currentCellCoords
.GetRow(); 
7541     int col 
= m_currentCellCoords
.GetCol(); 
7543     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
7546     wxRect rect 
= CellToRect(row
, col
); 
7548     // hmmm... what could we do here to show that the cell is disabled? 
7549     // for now, I just draw a thinner border than for the other ones, but 
7550     // it doesn't look really good 
7552     int penWidth 
= attr
->IsReadOnly() ? m_cellHighlightROPenWidth 
: m_cellHighlightPenWidth
; 
7556         // The center of the drawn line is where the position/width/height of 
7557         // the rectangle is actually at (on wxMSW at least), so the 
7558         // size of the rectangle is reduced to compensate for the thickness of 
7559         // the line. If this is too strange on non-wxMSW platforms then 
7560         // please #ifdef this appropriately. 
7561         rect
.x 
+= penWidth 
/ 2; 
7562         rect
.y 
+= penWidth 
/ 2; 
7563         rect
.width 
-= penWidth 
- 1; 
7564         rect
.height 
-= penWidth 
- 1; 
7566         // Now draw the rectangle 
7567         // use the cellHighlightColour if the cell is inside a selection, this 
7568         // will ensure the cell is always visible. 
7569         dc
.SetPen(wxPen(IsInSelection(row
,col
) ? m_selectionForeground 
: m_cellHighlightColour
, penWidth
, wxSOLID
)); 
7570         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
7571         dc
.DrawRectangle(rect
); 
7575         // VZ: my experiments with 3D borders... 
7577         // how to properly set colours for arbitrary bg? 
7578         wxCoord x1 
= rect
.x
, 
7580                 x2 
= rect
.x 
+ rect
.width 
- 1, 
7581                 y2 
= rect
.y 
+ rect
.height 
- 1; 
7583         dc
.SetPen(*wxWHITE_PEN
); 
7584         dc
.DrawLine(x1
, y1
, x2
, y1
); 
7585         dc
.DrawLine(x1
, y1
, x1
, y2
); 
7587         dc
.DrawLine(x1 
+ 1, y2 
- 1, x2 
- 1, y2 
- 1); 
7588         dc
.DrawLine(x2 
- 1, y1 
+ 1, x2 
- 1, y2
); 
7590         dc
.SetPen(*wxBLACK_PEN
); 
7591         dc
.DrawLine(x1
, y2
, x2
, y2
); 
7592         dc
.DrawLine(x2
, y1
, x2
, y2 
+ 1); 
7596 wxPen 
wxGrid::GetDefaultGridLinePen() 
7598     return wxPen(GetGridLineColour(), 1, wxSOLID
); 
7601 wxPen 
wxGrid::GetRowGridLinePen(int WXUNUSED(row
)) 
7603     return GetDefaultGridLinePen(); 
7606 wxPen 
wxGrid::GetColGridLinePen(int WXUNUSED(col
)) 
7608     return GetDefaultGridLinePen(); 
7611 void wxGrid::DrawCellBorder( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
7613     int row 
= coords
.GetRow(); 
7614     int col 
= coords
.GetCol(); 
7615     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
7619     wxRect rect 
= CellToRect( row
, col 
); 
7621     // right hand border 
7622     dc
.SetPen( GetColGridLinePen(col
) ); 
7623     dc
.DrawLine( rect
.x 
+ rect
.width
, rect
.y
, 
7624                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height 
+ 1 ); 
7627     dc
.SetPen( GetRowGridLinePen(row
) ); 
7628     dc
.DrawLine( rect
.x
, rect
.y 
+ rect
.height
, 
7629                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height
); 
7632 void wxGrid::DrawHighlight(wxDC
& dc
, const wxGridCellCoordsArray
& cells
) 
7634     // This if block was previously in wxGrid::OnPaint but that doesn't 
7635     // seem to get called under wxGTK - MB 
7637     if ( m_currentCellCoords 
== wxGridNoCellCoords 
&& 
7638          m_numRows 
&& m_numCols 
) 
7640         m_currentCellCoords
.Set(0, 0); 
7643     if ( IsCellEditControlShown() ) 
7645         // don't show highlight when the edit control is shown 
7649     // if the active cell was repainted, repaint its highlight too because it 
7650     // might have been damaged by the grid lines 
7651     size_t count 
= cells
.GetCount(); 
7652     for ( size_t n 
= 0; n 
< count
; n
++ ) 
7654         if ( cells
[n
] == m_currentCellCoords 
) 
7656             wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
7657             DrawCellHighlight(dc
, attr
); 
7665 // TODO: remove this ??? 
7666 // This is used to redraw all grid lines e.g. when the grid line colour 
7669 void wxGrid::DrawAllGridLines( wxDC
& dc
, const wxRegion 
& WXUNUSED(reg
) ) 
7671 #if !WXGRID_DRAW_LINES 
7675     if ( !m_gridLinesEnabled 
|| !m_numRows 
|| !m_numCols 
) 
7678     int top
, bottom
, left
, right
; 
7680 #if 0  //#ifndef __WXGTK__ 
7684       m_gridWin
->GetClientSize(&cw
, &ch
); 
7686       // virtual coords of visible area 
7688       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
7689       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
7694       reg
.GetBox(x
, y
, w
, h
); 
7695       CalcUnscrolledPosition( x
, y
, &left
, &top 
); 
7696       CalcUnscrolledPosition( x 
+ w
, y 
+ h
, &right
, &bottom 
); 
7700       m_gridWin
->GetClientSize(&cw
, &ch
); 
7701       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
7702       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
7705     // avoid drawing grid lines past the last row and col 
7707     right 
= wxMin( right
, GetColRight(GetColAt( m_numCols 
- 1 )) ); 
7708     bottom 
= wxMin( bottom
, GetRowBottom(m_numRows 
- 1) ); 
7710     // no gridlines inside multicells, clip them out 
7711     int leftCol 
= GetColPos( internalXToCol(left
) ); 
7712     int topRow 
= internalYToRow(top
); 
7713     int rightCol 
= GetColPos( internalXToCol(right
) ); 
7714     int bottomRow 
= internalYToRow(bottom
); 
7716 #if !defined(__WXMAC__) || wxMAC_USE_CORE_GRAPHICS 
7717     wxRegion 
clippedcells(0, 0, cw
, ch
); 
7719     int i
, j
, cell_rows
, cell_cols
; 
7722     for (j
=topRow
; j
<=bottomRow
; j
++) 
7725         for (colPos
=leftCol
; colPos
<=rightCol
; colPos
++) 
7727             i 
= GetColAt( colPos 
); 
7729             GetCellSize( j
, i
, &cell_rows
, &cell_cols 
); 
7730             if ((cell_rows 
> 1) || (cell_cols 
> 1)) 
7732                 rect 
= CellToRect(j
,i
); 
7733                 CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7734                 clippedcells
.Subtract(rect
); 
7736             else if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
7738                 rect 
= CellToRect(j 
+ cell_rows
, i 
+ cell_cols
); 
7739                 CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7740                 clippedcells
.Subtract(rect
); 
7745     wxRegion 
clippedcells( left
, top
, right 
- left
, bottom 
- top 
); 
7747     int i
, j
, cell_rows
, cell_cols
; 
7750     for (j
=topRow
; j
<=bottomRow
; j
++) 
7752         for (i
=leftCol
; i
<=rightCol
; i
++) 
7754             GetCellSize( j
, i
, &cell_rows
, &cell_cols 
); 
7755             if ((cell_rows 
> 1) || (cell_cols 
> 1)) 
7757                 rect 
= CellToRect(j
, i
); 
7758                 clippedcells
.Subtract(rect
); 
7760             else if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
7762                 rect 
= CellToRect(j 
+ cell_rows
, i 
+ cell_cols
); 
7763                 clippedcells
.Subtract(rect
); 
7769     dc
.SetClippingRegion( clippedcells 
); 
7772     // horizontal grid lines 
7774     // already declared above - int i; 
7775     for ( i 
= internalYToRow(top
); i 
< m_numRows
; i
++ ) 
7777         int bot 
= GetRowBottom(i
) - 1; 
7786             dc
.SetPen( GetRowGridLinePen(i
) ); 
7787             dc
.DrawLine( left
, bot
, right
, bot 
); 
7791     // vertical grid lines 
7794     for ( colPos 
= leftCol
; colPos 
< m_numCols
; colPos
++ ) 
7796         i 
= GetColAt( colPos 
); 
7798         int colRight 
= GetColRight(i
); 
7800         if (GetLayoutDirection() != wxLayout_RightToLeft
) 
7804         if ( colRight 
> right 
) 
7809         if ( colRight 
>= left 
) 
7811             dc
.SetPen( GetColGridLinePen(i
) ); 
7812             dc
.DrawLine( colRight
, top
, colRight
, bottom 
); 
7816     dc
.DestroyClippingRegion(); 
7819 void wxGrid::DrawRowLabels( wxDC
& dc
, const wxArrayInt
& rows
) 
7825     size_t numLabels 
= rows
.GetCount(); 
7827     for ( i 
= 0; i 
< numLabels
; i
++ ) 
7829         DrawRowLabel( dc
, rows
[i
] ); 
7833 void wxGrid::DrawRowLabel( wxDC
& dc
, int row 
) 
7835     if ( GetRowHeight(row
) <= 0 || m_rowLabelWidth 
<= 0 ) 
7843     rect
.SetY( GetRowTop(row
) + 1 ); 
7844     rect
.SetWidth( m_rowLabelWidth 
- 2 ); 
7845     rect
.SetHeight( GetRowHeight(row
) - 2 ); 
7847     CalcScrolledPosition( 0, rect
.y
, NULL
, &rect
.y 
); 
7849     wxWindowDC 
*win_dc 
= (wxWindowDC
*) &dc
; 
7851     wxRendererNative::Get().DrawHeaderButton( win_dc
->m_owner
, dc
, rect
, 0 ); 
7853     int rowTop 
= GetRowTop(row
), 
7854         rowBottom 
= GetRowBottom(row
) - 1; 
7856     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DSHADOW
), 1, wxSOLID
) ); 
7857     dc
.DrawLine( m_rowLabelWidth 
- 1, rowTop
, m_rowLabelWidth 
- 1, rowBottom 
); 
7858     dc
.DrawLine( 0, rowTop
, 0, rowBottom 
); 
7859     dc
.DrawLine( 0, rowBottom
, m_rowLabelWidth
, rowBottom 
); 
7861     dc
.SetPen( *wxWHITE_PEN 
); 
7862     dc
.DrawLine( 1, rowTop
, 1, rowBottom 
); 
7863     dc
.DrawLine( 1, rowTop
, m_rowLabelWidth 
- 1, rowTop 
); 
7866     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
7867     dc
.SetTextForeground( GetLabelTextColour() ); 
7868     dc
.SetFont( GetLabelFont() ); 
7871     GetRowLabelAlignment( &hAlign
, &vAlign 
); 
7874     rect
.SetY( GetRowTop(row
) + 2 ); 
7875     rect
.SetWidth( m_rowLabelWidth 
- 4 ); 
7876     rect
.SetHeight( GetRowHeight(row
) - 4 ); 
7877     DrawTextRectangle( dc
, GetRowLabelValue( row 
), rect
, hAlign
, vAlign 
); 
7880 void wxGrid::DrawColLabels( wxDC
& dc
,const wxArrayInt
& cols 
) 
7886     size_t numLabels 
= cols
.GetCount(); 
7888     for ( i 
= 0; i 
< numLabels
; i
++ ) 
7890         DrawColLabel( dc
, cols
[i
] ); 
7894 void wxGrid::DrawColLabel( wxDC
& dc
, int col 
) 
7896     if ( GetColWidth(col
) <= 0 || m_colLabelHeight 
<= 0 ) 
7899     int colLeft 
= GetColLeft(col
); 
7905     rect
.SetX( colLeft 
+ 1 ); 
7907     rect
.SetWidth( GetColWidth(col
) - 2 ); 
7908     rect
.SetHeight( m_colLabelHeight 
- 2 ); 
7910     wxWindowDC 
*win_dc 
= (wxWindowDC
*) &dc
; 
7912     wxRendererNative::Get().DrawHeaderButton( win_dc
->m_owner
, dc
, rect
, 0 ); 
7914     int colRight 
= GetColRight(col
) - 1; 
7916     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DSHADOW
), 1, wxSOLID
) ); 
7917     dc
.DrawLine( colRight
, 0, colRight
, m_colLabelHeight 
- 1 ); 
7918     dc
.DrawLine( colLeft
, 0, colRight
, 0 ); 
7919     dc
.DrawLine( colLeft
, m_colLabelHeight 
- 1, 
7920                  colRight 
+ 1, m_colLabelHeight 
- 1 ); 
7922     dc
.SetPen( *wxWHITE_PEN 
); 
7923     dc
.DrawLine( colLeft
, 1, colLeft
, m_colLabelHeight 
- 1 ); 
7924     dc
.DrawLine( colLeft
, 1, colRight
, 1 ); 
7927     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
7928     dc
.SetTextForeground( GetLabelTextColour() ); 
7929     dc
.SetFont( GetLabelFont() ); 
7931     int hAlign
, vAlign
, orient
; 
7932     GetColLabelAlignment( &hAlign
, &vAlign 
); 
7933     orient 
= GetColLabelTextOrientation(); 
7935     rect
.SetX( colLeft 
+ 2 ); 
7937     rect
.SetWidth( GetColWidth(col
) - 4 ); 
7938     rect
.SetHeight( m_colLabelHeight 
- 4 ); 
7939     DrawTextRectangle( dc
, GetColLabelValue( col 
), rect
, hAlign
, vAlign
, orient 
); 
7942 void wxGrid::DrawTextRectangle( wxDC
& dc
, 
7943                                 const wxString
& value
, 
7947                                 int textOrientation 
) 
7949     wxArrayString lines
; 
7951     StringToLines( value
, lines 
); 
7953     // Forward to new API. 
7954     DrawTextRectangle( dc
, 
7962 // VZ: this should be replaced with wxDC::DrawLabel() to which we just have to 
7963 //     add textOrientation support 
7964 void wxGrid::DrawTextRectangle(wxDC
& dc
, 
7965                                const wxArrayString
& lines
, 
7969                                int textOrientation
) 
7971     if ( lines
.empty() ) 
7974     wxDCClipper 
clip(dc
, rect
); 
7979     if ( textOrientation 
== wxHORIZONTAL 
) 
7980         GetTextBoxSize( dc
, lines
, &textWidth
, &textHeight 
); 
7982         GetTextBoxSize( dc
, lines
, &textHeight
, &textWidth 
); 
7986     switch ( vertAlign 
) 
7988         case wxALIGN_BOTTOM
: 
7989             if ( textOrientation 
== wxHORIZONTAL 
) 
7990                 y 
= rect
.y 
+ (rect
.height 
- textHeight 
- 1); 
7992                 x 
= rect
.x 
+ rect
.width 
- textWidth
; 
7995         case wxALIGN_CENTRE
: 
7996             if ( textOrientation 
== wxHORIZONTAL 
) 
7997                 y 
= rect
.y 
+ ((rect
.height 
- textHeight
) / 2); 
7999                 x 
= rect
.x 
+ ((rect
.width 
- textWidth
) / 2); 
8004             if ( textOrientation 
== wxHORIZONTAL 
) 
8011     // Align each line of a multi-line label 
8012     size_t nLines 
= lines
.GetCount(); 
8013     for ( size_t l 
= 0; l 
< nLines
; l
++ ) 
8015         const wxString
& line 
= lines
[l
]; 
8019             *(textOrientation 
== wxHORIZONTAL 
? &y 
: &x
) += dc
.GetCharHeight(); 
8025         dc
.GetTextExtent(line
, &lineWidth
, &lineHeight
); 
8027         switch ( horizAlign 
) 
8030                 if ( textOrientation 
== wxHORIZONTAL 
) 
8031                     x 
= rect
.x 
+ (rect
.width 
- lineWidth 
- 1); 
8033                     y 
= rect
.y 
+ lineWidth 
+ 1; 
8036             case wxALIGN_CENTRE
: 
8037                 if ( textOrientation 
== wxHORIZONTAL 
) 
8038                     x 
= rect
.x 
+ ((rect
.width 
- lineWidth
) / 2); 
8040                     y 
= rect
.y 
+ rect
.height 
- ((rect
.height 
- lineWidth
) / 2); 
8045                 if ( textOrientation 
== wxHORIZONTAL 
) 
8048                     y 
= rect
.y 
+ rect
.height 
- 1; 
8052         if ( textOrientation 
== wxHORIZONTAL 
) 
8054             dc
.DrawText( line
, x
, y 
); 
8059             dc
.DrawRotatedText( line
, x
, y
, 90.0 ); 
8065 // Split multi-line text up into an array of strings. 
8066 // Any existing contents of the string array are preserved. 
8068 void wxGrid::StringToLines( const wxString
& value
, wxArrayString
& lines 
) 
8072     wxString eol 
= wxTextFile::GetEOL( wxTextFileType_Unix 
); 
8073     wxString tVal 
= wxTextFile::Translate( value
, wxTextFileType_Unix 
); 
8075     while ( startPos 
< (int)tVal
.length() ) 
8077         pos 
= tVal
.Mid(startPos
).Find( eol 
); 
8082         else if ( pos 
== 0 ) 
8084             lines
.Add( wxEmptyString 
); 
8088             lines
.Add( value
.Mid(startPos
, pos
) ); 
8091         startPos 
+= pos 
+ 1; 
8094     if ( startPos 
< (int)value
.length() ) 
8096         lines
.Add( value
.Mid( startPos 
) ); 
8100 void wxGrid::GetTextBoxSize( const wxDC
& dc
, 
8101                              const wxArrayString
& lines
, 
8102                              long *width
, long *height 
) 
8106     long lineW 
= 0, lineH 
= 0; 
8109     for ( i 
= 0; i 
< lines
.GetCount(); i
++ ) 
8111         dc
.GetTextExtent( lines
[i
], &lineW
, &lineH 
); 
8112         w 
= wxMax( w
, lineW 
); 
8121 // ------ Batch processing. 
8123 void wxGrid::EndBatch() 
8125     if ( m_batchCount 
> 0 ) 
8128         if ( !m_batchCount 
) 
8131             m_rowLabelWin
->Refresh(); 
8132             m_colLabelWin
->Refresh(); 
8133             m_cornerLabelWin
->Refresh(); 
8134             m_gridWin
->Refresh(); 
8139 // Use this, rather than wxWindow::Refresh(), to force an immediate 
8140 // repainting of the grid. Has no effect if you are already inside a 
8141 // BeginBatch / EndBatch block. 
8143 void wxGrid::ForceRefresh() 
8149 bool wxGrid::Enable(bool enable
) 
8151     if ( !wxScrolledWindow::Enable(enable
) ) 
8154     // redraw in the new state 
8155     m_gridWin
->Refresh(); 
8161 // ------ Edit control functions 
8164 void wxGrid::EnableEditing( bool edit 
) 
8166     // TODO: improve this ? 
8168     if ( edit 
!= m_editable 
) 
8171             EnableCellEditControl(edit
); 
8176 void wxGrid::EnableCellEditControl( bool enable 
) 
8181     if ( enable 
!= m_cellEditCtrlEnabled 
) 
8185             if (SendEvent( wxEVT_GRID_EDITOR_SHOWN
) <0) 
8188             // this should be checked by the caller! 
8189             wxASSERT_MSG( CanEnableCellControl(), _T("can't enable editing for this cell!") ); 
8191             // do it before ShowCellEditControl() 
8192             m_cellEditCtrlEnabled 
= enable
; 
8194             ShowCellEditControl(); 
8198             //FIXME:add veto support 
8199             SendEvent( wxEVT_GRID_EDITOR_HIDDEN 
); 
8201             HideCellEditControl(); 
8202             SaveEditControlValue(); 
8204             // do it after HideCellEditControl() 
8205             m_cellEditCtrlEnabled 
= enable
; 
8210 bool wxGrid::IsCurrentCellReadOnly() const 
8213     wxGridCellAttr
* attr 
= ((wxGrid 
*)this)->GetCellAttr(m_currentCellCoords
); 
8214     bool readonly 
= attr
->IsReadOnly(); 
8220 bool wxGrid::CanEnableCellControl() const 
8222     return m_editable 
&& (m_currentCellCoords 
!= wxGridNoCellCoords
) && 
8223         !IsCurrentCellReadOnly(); 
8226 bool wxGrid::IsCellEditControlEnabled() const 
8228     // the cell edit control might be disable for all cells or just for the 
8229     // current one if it's read only 
8230     return m_cellEditCtrlEnabled 
? !IsCurrentCellReadOnly() : false; 
8233 bool wxGrid::IsCellEditControlShown() const 
8235     bool isShown 
= false; 
8237     if ( m_cellEditCtrlEnabled 
) 
8239         int row 
= m_currentCellCoords
.GetRow(); 
8240         int col 
= m_currentCellCoords
.GetCol(); 
8241         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8242         wxGridCellEditor
* editor 
= attr
->GetEditor((wxGrid
*) this, row
, col
); 
8247             if ( editor
->IsCreated() ) 
8249                 isShown 
= editor
->GetControl()->IsShown(); 
8259 void wxGrid::ShowCellEditControl() 
8261     if ( IsCellEditControlEnabled() ) 
8263         if ( !IsVisible( m_currentCellCoords
, false ) ) 
8265             m_cellEditCtrlEnabled 
= false; 
8270             wxRect rect 
= CellToRect( m_currentCellCoords 
); 
8271             int row 
= m_currentCellCoords
.GetRow(); 
8272             int col 
= m_currentCellCoords
.GetCol(); 
8274             // if this is part of a multicell, find owner (topleft) 
8275             int cell_rows
, cell_cols
; 
8276             GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
8277             if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
8281                 m_currentCellCoords
.SetRow( row 
); 
8282                 m_currentCellCoords
.SetCol( col 
); 
8285             // erase the highlight and the cell contents because the editor 
8286             // might not cover the entire cell 
8287             wxClientDC 
dc( m_gridWin 
); 
8289             dc
.SetBrush(wxBrush(GetCellAttr(row
, col
)->GetBackgroundColour(), wxSOLID
)); 
8290             dc
.SetPen(*wxTRANSPARENT_PEN
); 
8291             dc
.DrawRectangle(rect
); 
8293             // convert to scrolled coords 
8294             CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
8300             // cell is shifted by one pixel 
8301             // However, don't allow x or y to become negative 
8302             // since the SetSize() method interprets that as 
8309             wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8310             wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
8311             if ( !editor
->IsCreated() ) 
8313                 editor
->Create(m_gridWin
, wxID_ANY
, 
8314                                new wxGridCellEditorEvtHandler(this, editor
)); 
8316                 wxGridEditorCreatedEvent 
evt(GetId(), 
8317                                              wxEVT_GRID_EDITOR_CREATED
, 
8321                                              editor
->GetControl()); 
8322                 GetEventHandler()->ProcessEvent(evt
); 
8325             // resize editor to overflow into righthand cells if allowed 
8326             int maxWidth 
= rect
.width
; 
8327             wxString value 
= GetCellValue(row
, col
); 
8328             if ( (value 
!= wxEmptyString
) && (attr
->GetOverflow()) ) 
8331                 GetTextExtent(value
, &maxWidth
, &y
, NULL
, NULL
, &attr
->GetFont()); 
8332                 if (maxWidth 
< rect
.width
) 
8333                     maxWidth 
= rect
.width
; 
8336             int client_right 
= m_gridWin
->GetClientSize().GetWidth(); 
8337             if (rect
.x 
+ maxWidth 
> client_right
) 
8338                 maxWidth 
= client_right 
- rect
.x
; 
8340             if ((maxWidth 
> rect
.width
) && (col 
< m_numCols
) && m_table
) 
8342                 GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
8343                 // may have changed earlier 
8344                 for (int i 
= col 
+ cell_cols
; i 
< m_numCols
; i
++) 
8347                     GetCellSize( row
, i
, &c_rows
, &c_cols 
); 
8349                     // looks weird going over a multicell 
8350                     if (m_table
->IsEmptyCell( row
, i 
) && 
8351                             (rect
.width 
< maxWidth
) && (c_rows 
== 1)) 
8353                         rect
.width 
+= GetColWidth( i 
); 
8359                 if (rect
.GetRight() > client_right
) 
8360                     rect
.SetRight( client_right 
- 1 ); 
8363             editor
->SetCellAttr( attr 
); 
8364             editor
->SetSize( rect 
); 
8366                 editor
->GetControl()->Move( 
8367                     editor
->GetControl()->GetPosition().x 
+ nXMove
, 
8368                     editor
->GetControl()->GetPosition().y 
); 
8369             editor
->Show( true, attr 
); 
8371             // recalc dimensions in case we need to 
8372             // expand the scrolled window to account for editor 
8375             editor
->BeginEdit(row
, col
, this); 
8376             editor
->SetCellAttr(NULL
); 
8384 void wxGrid::HideCellEditControl() 
8386     if ( IsCellEditControlEnabled() ) 
8388         int row 
= m_currentCellCoords
.GetRow(); 
8389         int col 
= m_currentCellCoords
.GetCol(); 
8391         wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8392         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
8393         editor
->Show( false ); 
8397         m_gridWin
->SetFocus(); 
8399         // refresh whole row to the right 
8400         wxRect 
rect( CellToRect(row
, col
) ); 
8401         CalcScrolledPosition(rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
8402         rect
.width 
= m_gridWin
->GetClientSize().GetWidth() - rect
.x
; 
8405         // ensure that the pixels under the focus ring get refreshed as well 
8406         rect
.Inflate(10, 10); 
8409         m_gridWin
->Refresh( false, &rect 
); 
8413 void wxGrid::SaveEditControlValue() 
8415     if ( IsCellEditControlEnabled() ) 
8417         int row 
= m_currentCellCoords
.GetRow(); 
8418         int col 
= m_currentCellCoords
.GetCol(); 
8420         wxString oldval 
= GetCellValue(row
, col
); 
8422         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8423         wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
8424         bool changed 
= editor
->EndEdit(row
, col
, this); 
8431             if ( SendEvent( wxEVT_GRID_CELL_CHANGE
, 
8432                        m_currentCellCoords
.GetRow(), 
8433                        m_currentCellCoords
.GetCol() ) < 0 ) 
8435                 // Event has been vetoed, set the data back. 
8436                 SetCellValue(row
, col
, oldval
); 
8443 // ------ Grid location functions 
8444 //  Note that all of these functions work with the logical coordinates of 
8445 //  grid cells and labels so you will need to convert from device 
8446 //  coordinates for mouse events etc. 
8449 void wxGrid::XYToCell( int x
, int y
, wxGridCellCoords
& coords 
) 
8451     int row 
= YToRow(y
); 
8452     int col 
= XToCol(x
); 
8454     if ( row 
== -1 || col 
== -1 ) 
8456         coords 
= wxGridNoCellCoords
; 
8460         coords
.Set( row
, col 
); 
8464 // Internal Helper function for computing row or column from some 
8465 // (unscrolled) coordinate value, using either 
8466 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
8467 // of m_rowBottoms/m_ColRights to speed up the search! 
8469 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
8470                            const wxArrayInt
& BorderArray
, int nMax
, 
8474         return clipToMinMax 
&& (nMax 
> 0) ? 0 : -1; 
8479     size_t i_max 
= coord 
/ defaultDist
, 
8482     if (BorderArray
.IsEmpty()) 
8484         if ((int) i_max 
< nMax
) 
8486         return clipToMinMax 
? nMax 
- 1 : -1; 
8489     if ( i_max 
>= BorderArray
.GetCount()) 
8491         i_max 
= BorderArray
.GetCount() - 1; 
8495         if ( coord 
>= BorderArray
[i_max
]) 
8499                 i_max 
= coord 
/ minDist
; 
8501                 i_max 
=  BorderArray
.GetCount() - 1; 
8504         if ( i_max 
>= BorderArray
.GetCount()) 
8505             i_max 
= BorderArray
.GetCount() - 1; 
8508     if ( coord 
>= BorderArray
[i_max
]) 
8509         return clipToMinMax 
? (int)i_max 
: -1; 
8510     if ( coord 
< BorderArray
[0] ) 
8513     while ( i_max 
- i_min 
> 0 ) 
8515         wxCHECK_MSG(BorderArray
[i_min
] <= coord 
&& coord 
< BorderArray
[i_max
], 
8516                     0, _T("wxGrid: internal error in CoordToRowOrCol")); 
8517         if (coord 
>=  BorderArray
[ i_max 
- 1]) 
8521         int median 
= i_min 
+ (i_max 
- i_min 
+ 1) / 2; 
8522         if (coord 
< BorderArray
[median
]) 
8531 int wxGrid::YToRow( int y 
) 
8533     return CoordToRowOrCol(y
, m_defaultRowHeight
, 
8534                            m_minAcceptableRowHeight
, m_rowBottoms
, m_numRows
, false); 
8537 int wxGrid::XToCol( int x
, bool clipToMinMax 
) 
8540         return clipToMinMax 
&& (m_numCols 
> 0) ? GetColAt( 0 ) : -1; 
8542     if (!m_defaultColWidth
) 
8543         m_defaultColWidth 
= 1; 
8545     int maxPos 
= x 
/ m_defaultColWidth
; 
8548     if (m_colRights
.IsEmpty()) 
8550         if(maxPos 
< m_numCols
) 
8551             return GetColAt( maxPos 
); 
8552         return clipToMinMax 
? GetColAt( m_numCols 
- 1 ) : -1; 
8555     if ( maxPos 
>= m_numCols
) 
8556         maxPos 
= m_numCols 
- 1; 
8559         if ( x 
>= m_colRights
[GetColAt( maxPos 
)]) 
8562             if (m_minAcceptableColWidth
) 
8563                 maxPos 
= x 
/ m_minAcceptableColWidth
; 
8565                 maxPos 
=  m_numCols 
- 1; 
8567         if ( maxPos 
>= m_numCols
) 
8568             maxPos 
= m_numCols 
- 1; 
8571     //X is beyond the last column 
8572     if ( x 
>= m_colRights
[GetColAt( maxPos 
)]) 
8573         return clipToMinMax 
? GetColAt( maxPos 
) : -1; 
8575     //X is before the first column 
8576     if ( x 
< m_colRights
[GetColAt( 0 )] ) 
8577         return GetColAt( 0 ); 
8579     //Perform a binary search 
8580     while ( maxPos 
- minPos 
> 0 ) 
8582         wxCHECK_MSG(m_colRights
[GetColAt( minPos 
)] <= x 
&& x 
< m_colRights
[GetColAt( maxPos 
)], 
8583                     0, _T("wxGrid: internal error in XToCol")); 
8585         if (x 
>=  m_colRights
[GetColAt( maxPos 
- 1 )]) 
8586             return GetColAt( maxPos 
); 
8589         int median 
= minPos 
+ (maxPos 
- minPos 
+ 1) / 2; 
8590         if (x 
< m_colRights
[GetColAt( median 
)]) 
8595     return GetColAt( maxPos 
); 
8598 // return the row number that that the y coord is near 
8599 //  the edge of, or -1 if not near an edge. 
8600 // coords can only possibly be near an edge if 
8601 //    (a) the row/column is large enough to still allow for an "inner" area 
8602 //        that is _not_ nead the edge (i.e., if the height/width is smaller 
8603 //        than WXGRID_LABEL_EDGE_ZONE, coords are _never_ considered to be 
8606 //    (b) resizing rows/columns (the thing for which edge detection is 
8607 //        relevant at all) is enabled. 
8609 int wxGrid::YToEdgeOfRow( int y 
) 
8612     i 
= internalYToRow(y
); 
8614     if ( GetRowHeight(i
) > WXGRID_LABEL_EDGE_ZONE 
&& CanDragRowSize() ) 
8616         // We know that we are in row i, test whether we are 
8617         // close enough to lower or upper border, respectively. 
8618         if ( abs(GetRowBottom(i
) - y
) < WXGRID_LABEL_EDGE_ZONE 
) 
8620         else if ( i 
> 0 && y 
- GetRowTop(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
8627 // return the col number that that the x coord is near the edge of, or 
8628 // -1 if not near an edge 
8629 // See comment at YToEdgeOfRow for conditions on edge detection. 
8631 int wxGrid::XToEdgeOfCol( int x 
) 
8634     i 
= internalXToCol(x
); 
8636     if ( GetColWidth(i
) > WXGRID_LABEL_EDGE_ZONE 
&& CanDragColSize() ) 
8638         // We know that we are in column i; test whether we are 
8639         // close enough to right or left border, respectively. 
8640         if ( abs(GetColRight(i
) - x
) < WXGRID_LABEL_EDGE_ZONE 
) 
8642         else if ( i 
> 0 && x 
- GetColLeft(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
8649 wxRect 
wxGrid::CellToRect( int row
, int col 
) 
8651     wxRect 
rect( -1, -1, -1, -1 ); 
8653     if ( row 
>= 0 && row 
< m_numRows 
&& 
8654          col 
>= 0 && col 
< m_numCols 
) 
8656         int i
, cell_rows
, cell_cols
; 
8657         rect
.width 
= rect
.height 
= 0; 
8658         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
8659         // if negative then find multicell owner 
8664         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
8666         rect
.x 
= GetColLeft(col
); 
8667         rect
.y 
= GetRowTop(row
); 
8668         for (i
=col
; i 
< col 
+ cell_cols
; i
++) 
8669             rect
.width 
+= GetColWidth(i
); 
8670         for (i
=row
; i 
< row 
+ cell_rows
; i
++) 
8671             rect
.height 
+= GetRowHeight(i
); 
8674     // if grid lines are enabled, then the area of the cell is a bit smaller 
8675     if (m_gridLinesEnabled
) 
8684 bool wxGrid::IsVisible( int row
, int col
, bool wholeCellVisible 
) 
8686     // get the cell rectangle in logical coords 
8688     wxRect 
r( CellToRect( row
, col 
) ); 
8690     // convert to device coords 
8692     int left
, top
, right
, bottom
; 
8693     CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
8694     CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
8696     // check against the client area of the grid window 
8698     m_gridWin
->GetClientSize( &cw
, &ch 
); 
8700     if ( wholeCellVisible 
) 
8702         // is the cell wholly visible ? 
8703         return ( left 
>= 0 && right 
<= cw 
&& 
8704                  top 
>= 0 && bottom 
<= ch 
); 
8708         // is the cell partly visible ? 
8710         return ( ((left 
>= 0 && left 
< cw
) || (right 
> 0 && right 
<= cw
)) && 
8711                  ((top 
>= 0 && top 
< ch
) || (bottom 
> 0 && bottom 
<= ch
)) ); 
8715 // make the specified cell location visible by doing a minimal amount 
8718 void wxGrid::MakeCellVisible( int row
, int col 
) 
8721     int xpos 
= -1, ypos 
= -1; 
8723     if ( row 
>= 0 && row 
< m_numRows 
&& 
8724          col 
>= 0 && col 
< m_numCols 
) 
8726         // get the cell rectangle in logical coords 
8727         wxRect 
r( CellToRect( row
, col 
) ); 
8729         // convert to device coords 
8730         int left
, top
, right
, bottom
; 
8731         CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
8732         CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
8735         m_gridWin
->GetClientSize( &cw
, &ch 
); 
8741         else if ( bottom 
> ch 
) 
8743             int h 
= r
.GetHeight(); 
8745             for ( i 
= row 
- 1; i 
>= 0; i
-- ) 
8747                 int rowHeight 
= GetRowHeight(i
); 
8748                 if ( h 
+ rowHeight 
> ch 
) 
8755             // we divide it later by GRID_SCROLL_LINE, make sure that we don't 
8756             // have rounding errors (this is important, because if we do, 
8757             // we might not scroll at all and some cells won't be redrawn) 
8759             // Sometimes GRID_SCROLL_LINE / 2 is not enough, 
8760             // so just add a full scroll unit... 
8761             ypos 
+= m_scrollLineY
; 
8764         // special handling for wide cells - show always left part of the cell! 
8765         // Otherwise, e.g. when stepping from row to row, it would jump between 
8766         // left and right part of the cell on every step! 
8768         if ( left 
< 0 || (right 
- left
) >= cw 
) 
8772         else if ( right 
> cw 
) 
8774             // position the view so that the cell is on the right 
8776             CalcUnscrolledPosition(0, 0, &x0
, &y0
); 
8777             xpos 
= x0 
+ (right 
- cw
); 
8779             // see comment for ypos above 
8780             xpos 
+= m_scrollLineX
; 
8783         if ( xpos 
!= -1 || ypos 
!= -1 ) 
8786                 xpos 
/= m_scrollLineX
; 
8788                 ypos 
/= m_scrollLineY
; 
8789             Scroll( xpos
, ypos 
); 
8796 // ------ Grid cursor movement functions 
8799 bool wxGrid::MoveCursorUp( bool expandSelection 
) 
8801     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
8802          m_currentCellCoords
.GetRow() >= 0 ) 
8804         if ( expandSelection 
) 
8806             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
8807                 m_selectingKeyboard 
= m_currentCellCoords
; 
8808             if ( m_selectingKeyboard
.GetRow() > 0 ) 
8810                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() - 1 ); 
8811                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
8812                                  m_selectingKeyboard
.GetCol() ); 
8813                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8816         else if ( m_currentCellCoords
.GetRow() > 0 ) 
8818             int row 
= m_currentCellCoords
.GetRow() - 1; 
8819             int col 
= m_currentCellCoords
.GetCol(); 
8821             MakeCellVisible( row
, col 
); 
8822             SetCurrentCell( row
, col 
); 
8833 bool wxGrid::MoveCursorDown( bool expandSelection 
) 
8835     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
8836          m_currentCellCoords
.GetRow() < m_numRows 
) 
8838         if ( expandSelection 
) 
8840             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
8841                 m_selectingKeyboard 
= m_currentCellCoords
; 
8842             if ( m_selectingKeyboard
.GetRow() < m_numRows 
- 1 ) 
8844                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() + 1 ); 
8845                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
8846                         m_selectingKeyboard
.GetCol() ); 
8847                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8850         else if ( m_currentCellCoords
.GetRow() < m_numRows 
- 1 ) 
8852             int row 
= m_currentCellCoords
.GetRow() + 1; 
8853             int col 
= m_currentCellCoords
.GetCol(); 
8855             MakeCellVisible( row
, col 
); 
8856             SetCurrentCell( row
, col 
); 
8867 bool wxGrid::MoveCursorLeft( bool expandSelection 
) 
8869     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
8870          m_currentCellCoords
.GetCol() >= 0 ) 
8872         if ( expandSelection 
) 
8874             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
8875                 m_selectingKeyboard 
= m_currentCellCoords
; 
8876             if ( m_selectingKeyboard
.GetCol() > 0 ) 
8878                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() - 1 ); 
8879                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
8880                         m_selectingKeyboard
.GetCol() ); 
8881                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8884         else if ( GetColPos( m_currentCellCoords
.GetCol() ) > 0 ) 
8886             int row 
= m_currentCellCoords
.GetRow(); 
8887             int col 
= GetColAt( GetColPos( m_currentCellCoords
.GetCol() ) - 1 ); 
8890             MakeCellVisible( row
, col 
); 
8891             SetCurrentCell( row
, col 
); 
8902 bool wxGrid::MoveCursorRight( bool expandSelection 
) 
8904     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
8905          m_currentCellCoords
.GetCol() < m_numCols 
) 
8907         if ( expandSelection 
) 
8909             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
8910                 m_selectingKeyboard 
= m_currentCellCoords
; 
8911             if ( m_selectingKeyboard
.GetCol() < m_numCols 
- 1 ) 
8913                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() + 1 ); 
8914                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
8915                         m_selectingKeyboard
.GetCol() ); 
8916                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8919         else if ( GetColPos( m_currentCellCoords
.GetCol() ) < m_numCols 
- 1 ) 
8921             int row 
= m_currentCellCoords
.GetRow(); 
8922             int col 
= GetColAt( GetColPos( m_currentCellCoords
.GetCol() ) + 1 ); 
8925             MakeCellVisible( row
, col 
); 
8926             SetCurrentCell( row
, col 
); 
8937 bool wxGrid::MovePageUp() 
8939     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
8942     int row 
= m_currentCellCoords
.GetRow(); 
8946         m_gridWin
->GetClientSize( &cw
, &ch 
); 
8948         int y 
= GetRowTop(row
); 
8949         int newRow 
= internalYToRow( y 
- ch 
+ 1 ); 
8951         if ( newRow 
== row 
) 
8953             // row > 0, so newRow can never be less than 0 here. 
8957         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
8958         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
8966 bool wxGrid::MovePageDown() 
8968     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
8971     int row 
= m_currentCellCoords
.GetRow(); 
8972     if ( (row 
+ 1) < m_numRows 
) 
8975         m_gridWin
->GetClientSize( &cw
, &ch 
); 
8977         int y 
= GetRowTop(row
); 
8978         int newRow 
= internalYToRow( y 
+ ch 
); 
8979         if ( newRow 
== row 
) 
8981             // row < m_numRows, so newRow can't overflow here. 
8985         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
8986         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
8994 bool wxGrid::MoveCursorUpBlock( bool expandSelection 
) 
8997          m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
8998          m_currentCellCoords
.GetRow() > 0 ) 
9000         int row 
= m_currentCellCoords
.GetRow(); 
9001         int col 
= m_currentCellCoords
.GetCol(); 
9003         if ( m_table
->IsEmptyCell(row
, col
) ) 
9005             // starting in an empty cell: find the next block of 
9011                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9015         else if ( m_table
->IsEmptyCell(row 
- 1, col
) ) 
9017             // starting at the top of a block: find the next block 
9023                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9029             // starting within a block: find the top of the block 
9034                 if ( m_table
->IsEmptyCell(row
, col
) ) 
9042         MakeCellVisible( row
, col 
); 
9043         if ( expandSelection 
) 
9045             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
9046             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
9051             SetCurrentCell( row
, col 
); 
9060 bool wxGrid::MoveCursorDownBlock( bool expandSelection 
) 
9063          m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
9064          m_currentCellCoords
.GetRow() < m_numRows 
- 1 ) 
9066         int row 
= m_currentCellCoords
.GetRow(); 
9067         int col 
= m_currentCellCoords
.GetCol(); 
9069         if ( m_table
->IsEmptyCell(row
, col
) ) 
9071             // starting in an empty cell: find the next block of 
9074             while ( row 
< m_numRows 
- 1 ) 
9077                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9081         else if ( m_table
->IsEmptyCell(row 
+ 1, col
) ) 
9083             // starting at the bottom of a block: find the next block 
9086             while ( row 
< m_numRows 
- 1 ) 
9089                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9095             // starting within a block: find the bottom of the block 
9097             while ( row 
< m_numRows 
- 1 ) 
9100                 if ( m_table
->IsEmptyCell(row
, col
) ) 
9108         MakeCellVisible( row
, col 
); 
9109         if ( expandSelection 
) 
9111             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
9112             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
9117             SetCurrentCell( row
, col 
); 
9126 bool wxGrid::MoveCursorLeftBlock( bool expandSelection 
) 
9129          m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
9130          m_currentCellCoords
.GetCol() > 0 ) 
9132         int row 
= m_currentCellCoords
.GetRow(); 
9133         int col 
= m_currentCellCoords
.GetCol(); 
9135         if ( m_table
->IsEmptyCell(row
, col
) ) 
9137             // starting in an empty cell: find the next block of 
9143                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9147         else if ( m_table
->IsEmptyCell(row
, col 
- 1) ) 
9149             // starting at the left of a block: find the next block 
9155                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9161             // starting within a block: find the left of the block 
9166                 if ( m_table
->IsEmptyCell(row
, col
) ) 
9174         MakeCellVisible( row
, col 
); 
9175         if ( expandSelection 
) 
9177             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
9178             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
9183             SetCurrentCell( row
, col 
); 
9192 bool wxGrid::MoveCursorRightBlock( bool expandSelection 
) 
9195          m_currentCellCoords 
!= wxGridNoCellCoords 
&& 
9196          m_currentCellCoords
.GetCol() < m_numCols 
- 1 ) 
9198         int row 
= m_currentCellCoords
.GetRow(); 
9199         int col 
= m_currentCellCoords
.GetCol(); 
9201         if ( m_table
->IsEmptyCell(row
, col
) ) 
9203             // starting in an empty cell: find the next block of 
9206             while ( col 
< m_numCols 
- 1 ) 
9209                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9213         else if ( m_table
->IsEmptyCell(row
, col 
+ 1) ) 
9215             // starting at the right of a block: find the next block 
9218             while ( col 
< m_numCols 
- 1 ) 
9221                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) 
9227             // starting within a block: find the right of the block 
9229             while ( col 
< m_numCols 
- 1 ) 
9232                 if ( m_table
->IsEmptyCell(row
, col
) ) 
9240         MakeCellVisible( row
, col 
); 
9241         if ( expandSelection 
) 
9243             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
9244             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
9249             SetCurrentCell( row
, col 
); 
9259 // ------ Label values and formatting 
9262 void wxGrid::GetRowLabelAlignment( int *horiz
, int *vert 
) 
9265         *horiz 
= m_rowLabelHorizAlign
; 
9267         *vert  
= m_rowLabelVertAlign
; 
9270 void wxGrid::GetColLabelAlignment( int *horiz
, int *vert 
) 
9273         *horiz 
= m_colLabelHorizAlign
; 
9275         *vert  
= m_colLabelVertAlign
; 
9278 int wxGrid::GetColLabelTextOrientation() 
9280     return m_colLabelTextOrientation
; 
9283 wxString 
wxGrid::GetRowLabelValue( int row 
) 
9287         return m_table
->GetRowLabelValue( row 
); 
9297 wxString 
wxGrid::GetColLabelValue( int col 
) 
9301         return m_table
->GetColLabelValue( col 
); 
9311 void wxGrid::SetRowLabelSize( int width 
) 
9313     width 
= wxMax( width
, 0 ); 
9314     if ( width 
!= m_rowLabelWidth 
) 
9318             m_rowLabelWin
->Show( false ); 
9319             m_cornerLabelWin
->Show( false ); 
9321         else if ( m_rowLabelWidth 
== 0 ) 
9323             m_rowLabelWin
->Show( true ); 
9324             if ( m_colLabelHeight 
> 0 ) 
9325                 m_cornerLabelWin
->Show( true ); 
9328         m_rowLabelWidth 
= width
; 
9330         wxScrolledWindow::Refresh( true ); 
9334 void wxGrid::SetColLabelSize( int height 
) 
9336     height 
= wxMax( height
, 0 ); 
9337     if ( height 
!= m_colLabelHeight 
) 
9341             m_colLabelWin
->Show( false ); 
9342             m_cornerLabelWin
->Show( false ); 
9344         else if ( m_colLabelHeight 
== 0 ) 
9346             m_colLabelWin
->Show( true ); 
9347             if ( m_rowLabelWidth 
> 0 ) 
9348                 m_cornerLabelWin
->Show( true ); 
9351         m_colLabelHeight 
= height
; 
9353         wxScrolledWindow::Refresh( true ); 
9357 void wxGrid::SetLabelBackgroundColour( const wxColour
& colour 
) 
9359     if ( m_labelBackgroundColour 
!= colour 
) 
9361         m_labelBackgroundColour 
= colour
; 
9362         m_rowLabelWin
->SetBackgroundColour( colour 
); 
9363         m_colLabelWin
->SetBackgroundColour( colour 
); 
9364         m_cornerLabelWin
->SetBackgroundColour( colour 
); 
9366         if ( !GetBatchCount() ) 
9368             m_rowLabelWin
->Refresh(); 
9369             m_colLabelWin
->Refresh(); 
9370             m_cornerLabelWin
->Refresh(); 
9375 void wxGrid::SetLabelTextColour( const wxColour
& colour 
) 
9377     if ( m_labelTextColour 
!= colour 
) 
9379         m_labelTextColour 
= colour
; 
9380         if ( !GetBatchCount() ) 
9382             m_rowLabelWin
->Refresh(); 
9383             m_colLabelWin
->Refresh(); 
9388 void wxGrid::SetLabelFont( const wxFont
& font 
) 
9391     if ( !GetBatchCount() ) 
9393         m_rowLabelWin
->Refresh(); 
9394         m_colLabelWin
->Refresh(); 
9398 void wxGrid::SetRowLabelAlignment( int horiz
, int vert 
) 
9400     // allow old (incorrect) defs to be used 
9403         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
9404         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
9405         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
9410         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
9411         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
9412         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
9415     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
9417         m_rowLabelHorizAlign 
= horiz
; 
9420     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
9422         m_rowLabelVertAlign 
= vert
; 
9425     if ( !GetBatchCount() ) 
9427         m_rowLabelWin
->Refresh(); 
9431 void wxGrid::SetColLabelAlignment( int horiz
, int vert 
) 
9433     // allow old (incorrect) defs to be used 
9436         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
9437         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
9438         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
9443         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
9444         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
9445         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
9448     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
9450         m_colLabelHorizAlign 
= horiz
; 
9453     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
9455         m_colLabelVertAlign 
= vert
; 
9458     if ( !GetBatchCount() ) 
9460         m_colLabelWin
->Refresh(); 
9464 // Note: under MSW, the default column label font must be changed because it 
9465 //       does not support vertical printing 
9467 // Example: wxFont font(9, wxSWISS, wxNORMAL, wxBOLD); 
9468 //                      pGrid->SetLabelFont(font); 
9469 //                      pGrid->SetColLabelTextOrientation(wxVERTICAL); 
9471 void wxGrid::SetColLabelTextOrientation( int textOrientation 
) 
9473     if ( textOrientation 
== wxHORIZONTAL 
|| textOrientation 
== wxVERTICAL 
) 
9474         m_colLabelTextOrientation 
= textOrientation
; 
9476     if ( !GetBatchCount() ) 
9477         m_colLabelWin
->Refresh(); 
9480 void wxGrid::SetRowLabelValue( int row
, const wxString
& s 
) 
9484         m_table
->SetRowLabelValue( row
, s 
); 
9485         if ( !GetBatchCount() ) 
9487             wxRect rect 
= CellToRect( row
, 0 ); 
9488             if ( rect
.height 
> 0 ) 
9490                 CalcScrolledPosition(0, rect
.y
, &rect
.x
, &rect
.y
); 
9492                 rect
.width 
= m_rowLabelWidth
; 
9493                 m_rowLabelWin
->Refresh( true, &rect 
); 
9499 void wxGrid::SetColLabelValue( int col
, const wxString
& s 
) 
9503         m_table
->SetColLabelValue( col
, s 
); 
9504         if ( !GetBatchCount() ) 
9506             wxRect rect 
= CellToRect( 0, col 
); 
9507             if ( rect
.width 
> 0 ) 
9509                 CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &rect
.y
); 
9511                 rect
.height 
= m_colLabelHeight
; 
9512                 m_colLabelWin
->Refresh( true, &rect 
); 
9518 void wxGrid::SetGridLineColour( const wxColour
& colour 
) 
9520     if ( m_gridLineColour 
!= colour 
) 
9522         m_gridLineColour 
= colour
; 
9524         wxClientDC 
dc( m_gridWin 
); 
9526         DrawAllGridLines( dc
, wxRegion() ); 
9530 void wxGrid::SetCellHighlightColour( const wxColour
& colour 
) 
9532     if ( m_cellHighlightColour 
!= colour 
) 
9534         m_cellHighlightColour 
= colour
; 
9536         wxClientDC 
dc( m_gridWin 
); 
9538         wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
9539         DrawCellHighlight(dc
, attr
); 
9544 void wxGrid::SetCellHighlightPenWidth(int width
) 
9546     if (m_cellHighlightPenWidth 
!= width
) 
9548         m_cellHighlightPenWidth 
= width
; 
9550         // Just redrawing the cell highlight is not enough since that won't 
9551         // make any visible change if the the thickness is getting smaller. 
9552         int row 
= m_currentCellCoords
.GetRow(); 
9553         int col 
= m_currentCellCoords
.GetCol(); 
9554         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
9557         wxRect rect 
= CellToRect(row
, col
); 
9558         m_gridWin
->Refresh(true, &rect
); 
9562 void wxGrid::SetCellHighlightROPenWidth(int width
) 
9564     if (m_cellHighlightROPenWidth 
!= width
) 
9566         m_cellHighlightROPenWidth 
= width
; 
9568         // Just redrawing the cell highlight is not enough since that won't 
9569         // make any visible change if the the thickness is getting smaller. 
9570         int row 
= m_currentCellCoords
.GetRow(); 
9571         int col 
= m_currentCellCoords
.GetCol(); 
9572         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
9575         wxRect rect 
= CellToRect(row
, col
); 
9576         m_gridWin
->Refresh(true, &rect
); 
9580 void wxGrid::EnableGridLines( bool enable 
) 
9582     if ( enable 
!= m_gridLinesEnabled 
) 
9584         m_gridLinesEnabled 
= enable
; 
9586         if ( !GetBatchCount() ) 
9590                 wxClientDC 
dc( m_gridWin 
); 
9592                 DrawAllGridLines( dc
, wxRegion() ); 
9596                 m_gridWin
->Refresh(); 
9602 int wxGrid::GetDefaultRowSize() 
9604     return m_defaultRowHeight
; 
9607 int wxGrid::GetRowSize( int row 
) 
9609     wxCHECK_MSG( row 
>= 0 && row 
< m_numRows
, 0, _T("invalid row index") ); 
9611     return GetRowHeight(row
); 
9614 int wxGrid::GetDefaultColSize() 
9616     return m_defaultColWidth
; 
9619 int wxGrid::GetColSize( int col 
) 
9621     wxCHECK_MSG( col 
>= 0 && col 
< m_numCols
, 0, _T("invalid column index") ); 
9623     return GetColWidth(col
); 
9626 // ============================================================================ 
9627 // access to the grid attributes: each of them has a default value in the grid 
9628 // itself and may be overidden on a per-cell basis 
9629 // ============================================================================ 
9631 // ---------------------------------------------------------------------------- 
9632 // setting default attributes 
9633 // ---------------------------------------------------------------------------- 
9635 void wxGrid::SetDefaultCellBackgroundColour( const wxColour
& col 
) 
9637     m_defaultCellAttr
->SetBackgroundColour(col
); 
9639     m_gridWin
->SetBackgroundColour(col
); 
9643 void wxGrid::SetDefaultCellTextColour( const wxColour
& col 
) 
9645     m_defaultCellAttr
->SetTextColour(col
); 
9648 void wxGrid::SetDefaultCellAlignment( int horiz
, int vert 
) 
9650     m_defaultCellAttr
->SetAlignment(horiz
, vert
); 
9653 void wxGrid::SetDefaultCellOverflow( bool allow 
) 
9655     m_defaultCellAttr
->SetOverflow(allow
); 
9658 void wxGrid::SetDefaultCellFont( const wxFont
& font 
) 
9660     m_defaultCellAttr
->SetFont(font
); 
9663 // For editors and renderers the type registry takes precedence over the 
9664 // default attr, so we need to register the new editor/renderer for the string 
9665 // data type in order to make setting a default editor/renderer appear to 
9668 void wxGrid::SetDefaultRenderer(wxGridCellRenderer 
*renderer
) 
9670     RegisterDataType(wxGRID_VALUE_STRING
, 
9672                      GetDefaultEditorForType(wxGRID_VALUE_STRING
)); 
9675 void wxGrid::SetDefaultEditor(wxGridCellEditor 
*editor
) 
9677     RegisterDataType(wxGRID_VALUE_STRING
, 
9678                      GetDefaultRendererForType(wxGRID_VALUE_STRING
), 
9682 // ---------------------------------------------------------------------------- 
9683 // access to the default attrbiutes 
9684 // ---------------------------------------------------------------------------- 
9686 wxColour 
wxGrid::GetDefaultCellBackgroundColour() 
9688     return m_defaultCellAttr
->GetBackgroundColour(); 
9691 wxColour 
wxGrid::GetDefaultCellTextColour() 
9693     return m_defaultCellAttr
->GetTextColour(); 
9696 wxFont 
wxGrid::GetDefaultCellFont() 
9698     return m_defaultCellAttr
->GetFont(); 
9701 void wxGrid::GetDefaultCellAlignment( int *horiz
, int *vert 
) 
9703     m_defaultCellAttr
->GetAlignment(horiz
, vert
); 
9706 bool wxGrid::GetDefaultCellOverflow() 
9708     return m_defaultCellAttr
->GetOverflow(); 
9711 wxGridCellRenderer 
*wxGrid::GetDefaultRenderer() const 
9713     return m_defaultCellAttr
->GetRenderer(NULL
, 0, 0); 
9716 wxGridCellEditor 
*wxGrid::GetDefaultEditor() const 
9718     return m_defaultCellAttr
->GetEditor(NULL
, 0, 0); 
9721 // ---------------------------------------------------------------------------- 
9722 // access to cell attributes 
9723 // ---------------------------------------------------------------------------- 
9725 wxColour 
wxGrid::GetCellBackgroundColour(int row
, int col
) 
9727     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
9728     wxColour colour 
= attr
->GetBackgroundColour(); 
9734 wxColour 
wxGrid::GetCellTextColour( int row
, int col 
) 
9736     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
9737     wxColour colour 
= attr
->GetTextColour(); 
9743 wxFont 
wxGrid::GetCellFont( int row
, int col 
) 
9745     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
9746     wxFont font 
= attr
->GetFont(); 
9752 void wxGrid::GetCellAlignment( int row
, int col
, int *horiz
, int *vert 
) 
9754     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
9755     attr
->GetAlignment(horiz
, vert
); 
9759 bool wxGrid::GetCellOverflow( int row
, int col 
) 
9761     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
9762     bool allow 
= attr
->GetOverflow(); 
9768 void wxGrid::GetCellSize( int row
, int col
, int *num_rows
, int *num_cols 
) 
9770     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
9771     attr
->GetSize( num_rows
, num_cols 
); 
9775 wxGridCellRenderer
* wxGrid::GetCellRenderer(int row
, int col
) 
9777     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
9778     wxGridCellRenderer
* renderer 
= attr
->GetRenderer(this, row
, col
); 
9784 wxGridCellEditor
* wxGrid::GetCellEditor(int row
, int col
) 
9786     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
9787     wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
9793 bool wxGrid::IsReadOnly(int row
, int col
) const 
9795     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
9796     bool isReadOnly 
= attr
->IsReadOnly(); 
9802 // ---------------------------------------------------------------------------- 
9803 // attribute support: cache, automatic provider creation, ... 
9804 // ---------------------------------------------------------------------------- 
9806 bool wxGrid::CanHaveAttributes() 
9813     return m_table
->CanHaveAttributes(); 
9816 void wxGrid::ClearAttrCache() 
9818     if ( m_attrCache
.row 
!= -1 ) 
9820         wxSafeDecRef(m_attrCache
.attr
); 
9821         m_attrCache
.attr 
= NULL
; 
9822         m_attrCache
.row 
= -1; 
9826 void wxGrid::CacheAttr(int row
, int col
, wxGridCellAttr 
*attr
) const 
9830         wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
9832         self
->ClearAttrCache(); 
9833         self
->m_attrCache
.row 
= row
; 
9834         self
->m_attrCache
.col 
= col
; 
9835         self
->m_attrCache
.attr 
= attr
; 
9840 bool wxGrid::LookupAttr(int row
, int col
, wxGridCellAttr 
**attr
) const 
9842     if ( row 
== m_attrCache
.row 
&& col 
== m_attrCache
.col 
) 
9844         *attr 
= m_attrCache
.attr
; 
9845         wxSafeIncRef(m_attrCache
.attr
); 
9847 #ifdef DEBUG_ATTR_CACHE 
9848         gs_nAttrCacheHits
++; 
9855 #ifdef DEBUG_ATTR_CACHE 
9856         gs_nAttrCacheMisses
++; 
9863 wxGridCellAttr 
*wxGrid::GetCellAttr(int row
, int col
) const 
9865     wxGridCellAttr 
*attr 
= NULL
; 
9866     // Additional test to avoid looking at the cache e.g. for 
9867     // wxNoCellCoords, as this will confuse memory management. 
9870         if ( !LookupAttr(row
, col
, &attr
) ) 
9872             attr 
= m_table 
? m_table
->GetAttr(row
, col
, wxGridCellAttr::Any
) 
9873                            : (wxGridCellAttr 
*)NULL
; 
9874             CacheAttr(row
, col
, attr
); 
9880         attr
->SetDefAttr(m_defaultCellAttr
); 
9884         attr 
= m_defaultCellAttr
; 
9891 wxGridCellAttr 
*wxGrid::GetOrCreateCellAttr(int row
, int col
) const 
9893     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
9894     bool canHave 
= ((wxGrid
*)this)->CanHaveAttributes(); 
9896     wxCHECK_MSG( canHave
, attr
, _T("Cell attributes not allowed")); 
9897     wxCHECK_MSG( m_table
, attr
, _T("must have a table") ); 
9899     attr 
= m_table
->GetAttr(row
, col
, wxGridCellAttr::Cell
); 
9902         attr 
= new wxGridCellAttr(m_defaultCellAttr
); 
9904         // artificially inc the ref count to match DecRef() in caller 
9906         m_table
->SetAttr(attr
, row
, col
); 
9912 // ---------------------------------------------------------------------------- 
9913 // setting column attributes (wrappers around SetColAttr) 
9914 // ---------------------------------------------------------------------------- 
9916 void wxGrid::SetColFormatBool(int col
) 
9918     SetColFormatCustom(col
, wxGRID_VALUE_BOOL
); 
9921 void wxGrid::SetColFormatNumber(int col
) 
9923     SetColFormatCustom(col
, wxGRID_VALUE_NUMBER
); 
9926 void wxGrid::SetColFormatFloat(int col
, int width
, int precision
) 
9928     wxString typeName 
= wxGRID_VALUE_FLOAT
; 
9929     if ( (width 
!= -1) || (precision 
!= -1) ) 
9931         typeName 
<< _T(':') << width 
<< _T(',') << precision
; 
9934     SetColFormatCustom(col
, typeName
); 
9937 void wxGrid::SetColFormatCustom(int col
, const wxString
& typeName
) 
9939     wxGridCellAttr 
*attr 
= m_table
->GetAttr(-1, col
, wxGridCellAttr::Col 
); 
9941         attr 
= new wxGridCellAttr
; 
9942     wxGridCellRenderer 
*renderer 
= GetDefaultRendererForType(typeName
); 
9943     attr
->SetRenderer(renderer
); 
9945     SetColAttr(col
, attr
); 
9949 // ---------------------------------------------------------------------------- 
9950 // setting cell attributes: this is forwarded to the table 
9951 // ---------------------------------------------------------------------------- 
9953 void wxGrid::SetAttr(int row
, int col
, wxGridCellAttr 
*attr
) 
9955     if ( CanHaveAttributes() ) 
9957         m_table
->SetAttr(attr
, row
, col
); 
9966 void wxGrid::SetRowAttr(int row
, wxGridCellAttr 
*attr
) 
9968     if ( CanHaveAttributes() ) 
9970         m_table
->SetRowAttr(attr
, row
); 
9979 void wxGrid::SetColAttr(int col
, wxGridCellAttr 
*attr
) 
9981     if ( CanHaveAttributes() ) 
9983         m_table
->SetColAttr(attr
, col
); 
9992 void wxGrid::SetCellBackgroundColour( int row
, int col
, const wxColour
& colour 
) 
9994     if ( CanHaveAttributes() ) 
9996         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9997         attr
->SetBackgroundColour(colour
); 
10002 void wxGrid::SetCellTextColour( int row
, int col
, const wxColour
& colour 
) 
10004     if ( CanHaveAttributes() ) 
10006         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10007         attr
->SetTextColour(colour
); 
10012 void wxGrid::SetCellFont( int row
, int col
, const wxFont
& font 
) 
10014     if ( CanHaveAttributes() ) 
10016         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10017         attr
->SetFont(font
); 
10022 void wxGrid::SetCellAlignment( int row
, int col
, int horiz
, int vert 
) 
10024     if ( CanHaveAttributes() ) 
10026         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10027         attr
->SetAlignment(horiz
, vert
); 
10032 void wxGrid::SetCellOverflow( int row
, int col
, bool allow 
) 
10034     if ( CanHaveAttributes() ) 
10036         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10037         attr
->SetOverflow(allow
); 
10042 void wxGrid::SetCellSize( int row
, int col
, int num_rows
, int num_cols 
) 
10044     if ( CanHaveAttributes() ) 
10046         int cell_rows
, cell_cols
; 
10048         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10049         attr
->GetSize(&cell_rows
, &cell_cols
); 
10050         attr
->SetSize(num_rows
, num_cols
); 
10053         // Cannot set the size of a cell to 0 or negative values 
10054         // While it is perfectly legal to do that, this function cannot 
10055         // handle all the possibilies, do it by hand by getting the CellAttr. 
10056         // You can only set the size of a cell to 1,1 or greater with this fn 
10057         wxASSERT_MSG( !((cell_rows 
< 1) || (cell_cols 
< 1)), 
10058                       wxT("wxGrid::SetCellSize setting cell size that is already part of another cell")); 
10059         wxASSERT_MSG( !((num_rows 
< 1) || (num_cols 
< 1)), 
10060                       wxT("wxGrid::SetCellSize setting cell size to < 1")); 
10062         // if this was already a multicell then "turn off" the other cells first 
10063         if ((cell_rows 
> 1) || (cell_rows 
> 1)) 
10066             for (j
=row
; j 
< row 
+ cell_rows
; j
++) 
10068                 for (i
=col
; i 
< col 
+ cell_cols
; i
++) 
10070                     if ((i 
!= col
) || (j 
!= row
)) 
10072                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
10073                         attr_stub
->SetSize( 1, 1 ); 
10074                         attr_stub
->DecRef(); 
10080         // mark the cells that will be covered by this cell to 
10081         // negative or zero values to point back at this cell 
10082         if (((num_rows 
> 1) || (num_cols 
> 1)) && (num_rows 
>= 1) && (num_cols 
>= 1)) 
10085             for (j
=row
; j 
< row 
+ num_rows
; j
++) 
10087                 for (i
=col
; i 
< col 
+ num_cols
; i
++) 
10089                     if ((i 
!= col
) || (j 
!= row
)) 
10091                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
10092                         attr_stub
->SetSize( row 
- j
, col 
- i 
); 
10093                         attr_stub
->DecRef(); 
10101 void wxGrid::SetCellRenderer(int row
, int col
, wxGridCellRenderer 
*renderer
) 
10103     if ( CanHaveAttributes() ) 
10105         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10106         attr
->SetRenderer(renderer
); 
10111 void wxGrid::SetCellEditor(int row
, int col
, wxGridCellEditor
* editor
) 
10113     if ( CanHaveAttributes() ) 
10115         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10116         attr
->SetEditor(editor
); 
10121 void wxGrid::SetReadOnly(int row
, int col
, bool isReadOnly
) 
10123     if ( CanHaveAttributes() ) 
10125         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
10126         attr
->SetReadOnly(isReadOnly
); 
10131 // ---------------------------------------------------------------------------- 
10132 // Data type registration 
10133 // ---------------------------------------------------------------------------- 
10135 void wxGrid::RegisterDataType(const wxString
& typeName
, 
10136                               wxGridCellRenderer
* renderer
, 
10137                               wxGridCellEditor
* editor
) 
10139     m_typeRegistry
->RegisterDataType(typeName
, renderer
, editor
); 
10143 wxGridCellEditor 
* wxGrid::GetDefaultEditorForCell(int row
, int col
) const 
10145     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
10146     return GetDefaultEditorForType(typeName
); 
10149 wxGridCellRenderer 
* wxGrid::GetDefaultRendererForCell(int row
, int col
) const 
10151     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
10152     return GetDefaultRendererForType(typeName
); 
10155 wxGridCellEditor 
* wxGrid::GetDefaultEditorForType(const wxString
& typeName
) const 
10157     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
10158     if ( index 
== wxNOT_FOUND 
) 
10162         errStr
.Printf(wxT("Unknown data type name [%s]"), typeName
.c_str()); 
10163         wxFAIL_MSG(errStr
.c_str()); 
10168     return m_typeRegistry
->GetEditor(index
); 
10171 wxGridCellRenderer 
* wxGrid::GetDefaultRendererForType(const wxString
& typeName
) const 
10173     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
10174     if ( index 
== wxNOT_FOUND 
) 
10178         errStr
.Printf(wxT("Unknown data type name [%s]"), typeName
.c_str()); 
10179         wxFAIL_MSG(errStr
.c_str()); 
10184     return m_typeRegistry
->GetRenderer(index
); 
10187 // ---------------------------------------------------------------------------- 
10189 // ---------------------------------------------------------------------------- 
10191 void wxGrid::EnableDragRowSize( bool enable 
) 
10193     m_canDragRowSize 
= enable
; 
10196 void wxGrid::EnableDragColSize( bool enable 
) 
10198     m_canDragColSize 
= enable
; 
10201 void wxGrid::EnableDragGridSize( bool enable 
) 
10203     m_canDragGridSize 
= enable
; 
10206 void wxGrid::EnableDragCell( bool enable 
) 
10208     m_canDragCell 
= enable
; 
10211 void wxGrid::SetDefaultRowSize( int height
, bool resizeExistingRows 
) 
10213     m_defaultRowHeight 
= wxMax( height
, m_minAcceptableRowHeight 
); 
10215     if ( resizeExistingRows 
) 
10217         // since we are resizing all rows to the default row size, 
10218         // we can simply clear the row heights and row bottoms 
10219         // arrays (which also allows us to take advantage of 
10220         // some speed optimisations) 
10221         m_rowHeights
.Empty(); 
10222         m_rowBottoms
.Empty(); 
10223         if ( !GetBatchCount() ) 
10228 void wxGrid::SetRowSize( int row
, int height 
) 
10230     wxCHECK_RET( row 
>= 0 && row 
< m_numRows
, _T("invalid row index") ); 
10232     // See comment in SetColSize 
10233     if ( height 
< GetRowMinimalAcceptableHeight()) 
10236     if ( m_rowHeights
.IsEmpty() ) 
10238         // need to really create the array 
10242     int h 
= wxMax( 0, height 
); 
10243     int diff 
= h 
- m_rowHeights
[row
]; 
10245     m_rowHeights
[row
] = h
; 
10247     for ( i 
= row
; i 
< m_numRows
; i
++ ) 
10249         m_rowBottoms
[i
] += diff
; 
10252     if ( !GetBatchCount() ) 
10256 void wxGrid::SetDefaultColSize( int width
, bool resizeExistingCols 
) 
10258     m_defaultColWidth 
= wxMax( width
, m_minAcceptableColWidth 
); 
10260     if ( resizeExistingCols 
) 
10262         // since we are resizing all columns to the default column size, 
10263         // we can simply clear the col widths and col rights 
10264         // arrays (which also allows us to take advantage of 
10265         // some speed optimisations) 
10266         m_colWidths
.Empty(); 
10267         m_colRights
.Empty(); 
10268         if ( !GetBatchCount() ) 
10273 void wxGrid::SetColSize( int col
, int width 
) 
10275     wxCHECK_RET( col 
>= 0 && col 
< m_numCols
, _T("invalid column index") ); 
10277     // should we check that it's bigger than GetColMinimalWidth(col) here? 
10279     // No, because it is reasonable to assume the library user know's 
10280     // what he is doing. However we should test against the weaker 
10281     // constraint of minimalAcceptableWidth, as this breaks rendering 
10283     // This test then fixes sf.net bug #645734 
10285     if ( width 
< GetColMinimalAcceptableWidth() ) 
10288     if ( m_colWidths
.IsEmpty() ) 
10290         // need to really create the array 
10294     // if < 0 then calculate new width from label 
10298         wxArrayString lines
; 
10299         wxClientDC 
dc(m_colLabelWin
); 
10300         dc
.SetFont(GetLabelFont()); 
10301         StringToLines(GetColLabelValue(col
), lines
); 
10302         GetTextBoxSize(dc
, lines
, &w
, &h
); 
10306     int w 
= wxMax( 0, width 
); 
10307     int diff 
= w 
- m_colWidths
[col
]; 
10308     m_colWidths
[col
] = w
; 
10312     for ( colPos 
= GetColPos( col 
); colPos 
< m_numCols
; colPos
++ ) 
10314         i 
= GetColAt( colPos 
); 
10315         m_colRights
[i
] += diff
; 
10318     if ( !GetBatchCount() ) 
10322 void wxGrid::SetColMinimalWidth( int col
, int width 
) 
10324     if (width 
> GetColMinimalAcceptableWidth()) 
10326         wxLongToLongHashMap::key_type key 
= (wxLongToLongHashMap::key_type
)col
; 
10327         m_colMinWidths
[key
] = width
; 
10331 void wxGrid::SetRowMinimalHeight( int row
, int width 
) 
10333     if (width 
> GetRowMinimalAcceptableHeight()) 
10335         wxLongToLongHashMap::key_type key 
= (wxLongToLongHashMap::key_type
)row
; 
10336         m_rowMinHeights
[key
] = width
; 
10340 int wxGrid::GetColMinimalWidth(int col
) const 
10342     wxLongToLongHashMap::key_type key 
= (wxLongToLongHashMap::key_type
)col
; 
10343     wxLongToLongHashMap::const_iterator it 
= m_colMinWidths
.find(key
); 
10345     return it 
!= m_colMinWidths
.end() ? (int)it
->second 
: m_minAcceptableColWidth
; 
10348 int wxGrid::GetRowMinimalHeight(int row
) const 
10350     wxLongToLongHashMap::key_type key 
= (wxLongToLongHashMap::key_type
)row
; 
10351     wxLongToLongHashMap::const_iterator it 
= m_rowMinHeights
.find(key
); 
10353     return it 
!= m_rowMinHeights
.end() ? (int)it
->second 
: m_minAcceptableRowHeight
; 
10356 void wxGrid::SetColMinimalAcceptableWidth( int width 
) 
10358     // We do allow a width of 0 since this gives us 
10359     // an easy way to temporarily hiding columns. 
10361         m_minAcceptableColWidth 
= width
; 
10364 void wxGrid::SetRowMinimalAcceptableHeight( int height 
) 
10366     // We do allow a height of 0 since this gives us 
10367     // an easy way to temporarily hiding rows. 
10369         m_minAcceptableRowHeight 
= height
; 
10372 int  wxGrid::GetColMinimalAcceptableWidth() const 
10374     return m_minAcceptableColWidth
; 
10377 int  wxGrid::GetRowMinimalAcceptableHeight() const 
10379     return m_minAcceptableRowHeight
; 
10382 // ---------------------------------------------------------------------------- 
10384 // ---------------------------------------------------------------------------- 
10386 void wxGrid::AutoSizeColOrRow( int colOrRow
, bool setAsMin
, bool column 
) 
10388     wxClientDC 
dc(m_gridWin
); 
10390     // cancel editing of cell 
10391     HideCellEditControl(); 
10392     SaveEditControlValue(); 
10394     // init both of them to avoid compiler warnings, even if we only need one 
10402     wxCoord extent
, extentMax 
= 0; 
10403     int max 
= column 
? m_numRows 
: m_numCols
; 
10404     for ( int rowOrCol 
= 0; rowOrCol 
< max
; rowOrCol
++ ) 
10411         wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
10412         wxGridCellRenderer 
*renderer 
= attr
->GetRenderer(this, row
, col
); 
10415             wxSize size 
= renderer
->GetBestSize(*this, *attr
, dc
, row
, col
); 
10416             extent 
= column 
? size
.x 
: size
.y
; 
10417             if ( extent 
> extentMax 
) 
10418                 extentMax 
= extent
; 
10420             renderer
->DecRef(); 
10426     // now also compare with the column label extent 
10428     dc
.SetFont( GetLabelFont() ); 
10432         dc
.GetMultiLineTextExtent( GetColLabelValue(col
), &w
, &h 
); 
10433         if ( GetColLabelTextOrientation() == wxVERTICAL 
) 
10437         dc
.GetMultiLineTextExtent( GetRowLabelValue(row
), &w
, &h 
); 
10439     extent 
= column 
? w 
: h
; 
10440     if ( extent 
> extentMax 
) 
10441         extentMax 
= extent
; 
10445         // empty column - give default extent (notice that if extentMax is less 
10446         // than default extent but != 0, it's OK) 
10447         extentMax 
= column 
? m_defaultColWidth 
: m_defaultRowHeight
; 
10452             // leave some space around text 
10460         SetColSize( col
, extentMax 
); 
10461         if ( !GetBatchCount() ) 
10464             m_gridWin
->GetClientSize( &cw
, &ch 
); 
10465             wxRect 
rect ( CellToRect( 0, col 
) ); 
10467             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
10468             rect
.width 
= cw 
- rect
.x
; 
10469             rect
.height 
= m_colLabelHeight
; 
10470             m_colLabelWin
->Refresh( true, &rect 
); 
10475         SetRowSize(row
, extentMax
); 
10476         if ( !GetBatchCount() ) 
10479             m_gridWin
->GetClientSize( &cw
, &ch 
); 
10480             wxRect 
rect( CellToRect( row
, 0 ) ); 
10482             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
10483             rect
.width 
= m_rowLabelWidth
; 
10484             rect
.height 
= ch 
- rect
.y
; 
10485             m_rowLabelWin
->Refresh( true, &rect 
); 
10492             SetColMinimalWidth(col
, extentMax
); 
10494             SetRowMinimalHeight(row
, extentMax
); 
10498 int wxGrid::SetOrCalcColumnSizes(bool calcOnly
, bool setAsMin
) 
10500     int width 
= m_rowLabelWidth
; 
10505     for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
10508             AutoSizeColumn(col
, setAsMin
); 
10510         width 
+= GetColWidth(col
); 
10519 int wxGrid::SetOrCalcRowSizes(bool calcOnly
, bool setAsMin
) 
10521     int height 
= m_colLabelHeight
; 
10526     for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
10529             AutoSizeRow(row
, setAsMin
); 
10531         height 
+= GetRowHeight(row
); 
10540 void wxGrid::AutoSize() 
10544     wxSize 
size(SetOrCalcColumnSizes(false), SetOrCalcRowSizes(false)); 
10546     // round up the size to a multiple of scroll step - this ensures that we 
10547     // won't get the scrollbars if we're sized exactly to this width 
10548     // CalcDimension adds m_extraWidth + 1 etc. to calculate the necessary 
10551         GetScrollX(size
.x 
+ m_extraWidth 
+ 1) * m_scrollLineX
, 
10552         GetScrollY(size
.y 
+ m_extraHeight 
+ 1) * m_scrollLineY 
); 
10554     // distribute the extra space between the columns/rows to avoid having 
10555     // extra white space 
10557     // Remove the extra m_extraWidth + 1 added above 
10558     wxCoord diff 
= sizeFit
.x 
- size
.x 
+ (m_extraWidth 
+ 1); 
10559     if ( diff 
&& m_numCols 
) 
10561         // try to resize the columns uniformly 
10562         wxCoord diffPerCol 
= diff 
/ m_numCols
; 
10565             for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
10567                 SetColSize(col
, GetColWidth(col
) + diffPerCol
); 
10571         // add remaining amount to the last columns 
10572         diff 
-= diffPerCol 
* m_numCols
; 
10575             for ( int col 
= m_numCols 
- 1; col 
>= m_numCols 
- diff
; col
-- ) 
10577                 SetColSize(col
, GetColWidth(col
) + 1); 
10583     diff 
= sizeFit
.y 
- size
.y 
- (m_extraHeight 
+ 1); 
10584     if ( diff 
&& m_numRows 
) 
10586         // try to resize the columns uniformly 
10587         wxCoord diffPerRow 
= diff 
/ m_numRows
; 
10590             for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
10592                 SetRowSize(row
, GetRowHeight(row
) + diffPerRow
); 
10596         // add remaining amount to the last rows 
10597         diff 
-= diffPerRow 
* m_numRows
; 
10600             for ( int row 
= m_numRows 
- 1; row 
>= m_numRows 
- diff
; row
-- ) 
10602                 SetRowSize(row
, GetRowHeight(row
) + 1); 
10609     SetClientSize(sizeFit
); 
10612 void wxGrid::AutoSizeRowLabelSize( int row 
) 
10614     wxArrayString lines
; 
10617     // Hide the edit control, so it 
10618     // won't interfere with drag-shrinking. 
10619     if ( IsCellEditControlShown() ) 
10621         HideCellEditControl(); 
10622         SaveEditControlValue(); 
10625     // autosize row height depending on label text 
10626     StringToLines( GetRowLabelValue( row 
), lines 
); 
10627     wxClientDC 
dc( m_rowLabelWin 
); 
10628     GetTextBoxSize( dc
, lines
, &w
, &h 
); 
10629     if ( h 
< m_defaultRowHeight 
) 
10630         h 
= m_defaultRowHeight
; 
10631     SetRowSize(row
, h
); 
10635 void wxGrid::AutoSizeColLabelSize( int col 
) 
10637     wxArrayString lines
; 
10640     // Hide the edit control, so it 
10641     // won't interfere with drag-shrinking. 
10642     if ( IsCellEditControlShown() ) 
10644         HideCellEditControl(); 
10645         SaveEditControlValue(); 
10648     // autosize column width depending on label text 
10649     StringToLines( GetColLabelValue( col 
), lines 
); 
10650     wxClientDC 
dc( m_colLabelWin 
); 
10651     if ( GetColLabelTextOrientation() == wxHORIZONTAL 
) 
10652         GetTextBoxSize( dc
, lines
, &w
, &h 
); 
10654         GetTextBoxSize( dc
, lines
, &h
, &w 
); 
10655     if ( w 
< m_defaultColWidth 
) 
10656         w 
= m_defaultColWidth
; 
10657     SetColSize(col
, w
); 
10661 wxSize 
wxGrid::DoGetBestSize() const 
10663     // don't set sizes, only calculate them 
10664     wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
10667     width 
= self
->SetOrCalcColumnSizes(true); 
10668     height 
= self
->SetOrCalcRowSizes(true); 
10675     // Round up to a multiple the scroll rate 
10676     // NOTE: this still doesn't get rid of the scrollbars; 
10677     // is there any magic incantation for that? 
10679     GetScrollPixelsPerUnit(&xpu
, &ypu
); 
10681         width  
+= 1 + xpu 
- (width  
% xpu
); 
10683         height 
+= 1 + ypu 
- (height 
% ypu
); 
10685     // limit to 1/4 of the screen size 
10686     int maxwidth
, maxheight
; 
10687     wxDisplaySize( &maxwidth
, &maxheight 
); 
10690     if ( width 
> maxwidth 
) 
10692     if ( height 
> maxheight 
) 
10693         height 
= maxheight
; 
10695     wxSize 
best(width
, height
); 
10697     // NOTE: This size should be cached, but first we need to add calls to 
10698     // InvalidateBestSize everywhere that could change the results of this 
10700     // CacheBestSize(size); 
10710 wxPen
& wxGrid::GetDividerPen() const 
10715 // ---------------------------------------------------------------------------- 
10716 // cell value accessor functions 
10717 // ---------------------------------------------------------------------------- 
10719 void wxGrid::SetCellValue( int row
, int col
, const wxString
& s 
) 
10723         m_table
->SetValue( row
, col
, s 
); 
10724         if ( !GetBatchCount() ) 
10727             wxRect 
rect( CellToRect( row
, col 
) ); 
10729             rect
.width 
= m_gridWin
->GetClientSize().GetWidth(); 
10730             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
10731             m_gridWin
->Refresh( false, &rect 
); 
10734         if ( m_currentCellCoords
.GetRow() == row 
&& 
10735              m_currentCellCoords
.GetCol() == col 
&& 
10736              IsCellEditControlShown()) 
10737              // Note: If we are using IsCellEditControlEnabled, 
10738              // this interacts badly with calling SetCellValue from 
10739              // an EVT_GRID_CELL_CHANGE handler. 
10741             HideCellEditControl(); 
10742             ShowCellEditControl(); // will reread data from table 
10747 // ---------------------------------------------------------------------------- 
10748 // block, row and column selection 
10749 // ---------------------------------------------------------------------------- 
10751 void wxGrid::SelectRow( int row
, bool addToSelected 
) 
10753     if ( IsSelection() && !addToSelected 
) 
10757         m_selection
->SelectRow( row
, false, addToSelected 
); 
10760 void wxGrid::SelectCol( int col
, bool addToSelected 
) 
10762     if ( IsSelection() && !addToSelected 
) 
10766         m_selection
->SelectCol( col
, false, addToSelected 
); 
10769 void wxGrid::SelectBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol
, 
10770                           bool addToSelected 
) 
10772     if ( IsSelection() && !addToSelected 
) 
10776         m_selection
->SelectBlock( topRow
, leftCol
, bottomRow
, rightCol
, 
10777                                   false, addToSelected 
); 
10780 void wxGrid::SelectAll() 
10782     if ( m_numRows 
> 0 && m_numCols 
> 0 ) 
10785             m_selection
->SelectBlock( 0, 0, m_numRows 
- 1, m_numCols 
- 1 ); 
10789 // ---------------------------------------------------------------------------- 
10790 // cell, row and col deselection 
10791 // ---------------------------------------------------------------------------- 
10793 void wxGrid::DeselectRow( int row 
) 
10795     if ( !m_selection 
) 
10798     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
10800         if ( m_selection
->IsInSelection(row
, 0 ) ) 
10801             m_selection
->ToggleCellSelection(row
, 0); 
10805         int nCols 
= GetNumberCols(); 
10806         for ( int i 
= 0; i 
< nCols
; i
++ ) 
10808             if ( m_selection
->IsInSelection(row
, i 
) ) 
10809                 m_selection
->ToggleCellSelection(row
, i
); 
10814 void wxGrid::DeselectCol( int col 
) 
10816     if ( !m_selection 
) 
10819     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
10821         if ( m_selection
->IsInSelection(0, col 
) ) 
10822             m_selection
->ToggleCellSelection(0, col
); 
10826         int nRows 
= GetNumberRows(); 
10827         for ( int i 
= 0; i 
< nRows
; i
++ ) 
10829             if ( m_selection
->IsInSelection(i
, col 
) ) 
10830                 m_selection
->ToggleCellSelection(i
, col
); 
10835 void wxGrid::DeselectCell( int row
, int col 
) 
10837     if ( m_selection 
&& m_selection
->IsInSelection(row
, col
) ) 
10838         m_selection
->ToggleCellSelection(row
, col
); 
10841 bool wxGrid::IsSelection() 
10843     return ( m_selection 
&& (m_selection
->IsSelection() || 
10844              ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
10845                m_selectingBottomRight 
!= wxGridNoCellCoords
) ) ); 
10848 bool wxGrid::IsInSelection( int row
, int col 
) const 
10850     return ( m_selection 
&& (m_selection
->IsInSelection( row
, col 
) || 
10851              ( row 
>= m_selectingTopLeft
.GetRow() && 
10852                col 
>= m_selectingTopLeft
.GetCol() && 
10853                row 
<= m_selectingBottomRight
.GetRow() && 
10854                col 
<= m_selectingBottomRight
.GetCol() )) ); 
10857 wxGridCellCoordsArray 
wxGrid::GetSelectedCells() const 
10861         wxGridCellCoordsArray a
; 
10865     return m_selection
->m_cellSelection
; 
10868 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockTopLeft() const 
10872         wxGridCellCoordsArray a
; 
10876     return m_selection
->m_blockSelectionTopLeft
; 
10879 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockBottomRight() const 
10883         wxGridCellCoordsArray a
; 
10887     return m_selection
->m_blockSelectionBottomRight
; 
10890 wxArrayInt 
wxGrid::GetSelectedRows() const 
10898     return m_selection
->m_rowSelection
; 
10901 wxArrayInt 
wxGrid::GetSelectedCols() const 
10909     return m_selection
->m_colSelection
; 
10912 void wxGrid::ClearSelection() 
10914     m_selectingTopLeft 
= 
10915     m_selectingBottomRight 
= 
10916     m_selectingKeyboard 
= wxGridNoCellCoords
; 
10918         m_selection
->ClearSelection(); 
10921 // This function returns the rectangle that encloses the given block 
10922 // in device coords clipped to the client size of the grid window. 
10924 wxRect 
wxGrid::BlockToDeviceRect( const wxGridCellCoords 
&topLeft
, 
10925                                   const wxGridCellCoords 
&bottomRight 
) 
10927     wxRect 
rect( wxGridNoCellRect 
); 
10930     cellRect 
= CellToRect( topLeft 
); 
10931     if ( cellRect 
!= wxGridNoCellRect 
) 
10937         rect 
= wxRect(0, 0, 0, 0); 
10940     cellRect 
= CellToRect( bottomRight 
); 
10941     if ( cellRect 
!= wxGridNoCellRect 
) 
10947         return wxGridNoCellRect
; 
10951     int left 
= rect
.GetLeft(); 
10952     int top 
= rect
.GetTop(); 
10953     int right 
= rect
.GetRight(); 
10954     int bottom 
= rect
.GetBottom(); 
10956     int leftCol 
= topLeft
.GetCol(); 
10957     int topRow 
= topLeft
.GetRow(); 
10958     int rightCol 
= bottomRight
.GetCol(); 
10959     int bottomRow 
= bottomRight
.GetRow(); 
10967         leftCol 
= rightCol
; 
10977         topRow 
= bottomRow
; 
10981     for ( j 
= topRow
; j 
<= bottomRow
; j
++ ) 
10983         for ( i 
= leftCol
; i 
<= rightCol
; i
++ ) 
10985             if ((j 
== topRow
) || (j 
== bottomRow
) || (i 
== leftCol
) || (i 
== rightCol
)) 
10987                 cellRect 
= CellToRect( j
, i 
); 
10989                 if (cellRect
.x 
< left
) 
10991                 if (cellRect
.y 
< top
) 
10993                 if (cellRect
.x 
+ cellRect
.width 
> right
) 
10994                     right 
= cellRect
.x 
+ cellRect
.width
; 
10995                 if (cellRect
.y 
+ cellRect
.height 
> bottom
) 
10996                     bottom 
= cellRect
.y 
+ cellRect
.height
; 
11000                 i 
= rightCol
; // jump over inner cells. 
11005     // convert to scrolled coords 
11007     CalcScrolledPosition( left
, top
, &left
, &top 
); 
11008     CalcScrolledPosition( right
, bottom
, &right
, &bottom 
); 
11011     m_gridWin
->GetClientSize( &cw
, &ch 
); 
11013     if (right 
< 0 || bottom 
< 0 || left 
> cw 
|| top 
> ch
) 
11014         return wxRect(0,0,0,0); 
11016     rect
.SetLeft( wxMax(0, left
) ); 
11017     rect
.SetTop( wxMax(0, top
) ); 
11018     rect
.SetRight( wxMin(cw
, right
) ); 
11019     rect
.SetBottom( wxMin(ch
, bottom
) ); 
11024 // ---------------------------------------------------------------------------- 
11025 // grid event classes 
11026 // ---------------------------------------------------------------------------- 
11028 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent
, wxNotifyEvent 
) 
11030 wxGridEvent::wxGridEvent( int id
, wxEventType type
, wxObject
* obj
, 
11031                           int row
, int col
, int x
, int y
, bool sel
, 
11032                           bool control
, bool shift
, bool alt
, bool meta 
) 
11033         : wxNotifyEvent( type
, id 
) 
11040     m_control 
= control
; 
11045     SetEventObject(obj
); 
11049 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent
, wxNotifyEvent 
) 
11051 wxGridSizeEvent::wxGridSizeEvent( int id
, wxEventType type
, wxObject
* obj
, 
11052                                   int rowOrCol
, int x
, int y
, 
11053                                   bool control
, bool shift
, bool alt
, bool meta 
) 
11054         : wxNotifyEvent( type
, id 
) 
11056     m_rowOrCol 
= rowOrCol
; 
11059     m_control 
= control
; 
11064     SetEventObject(obj
); 
11068 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent
, wxNotifyEvent 
) 
11070 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id
, wxEventType type
, wxObject
* obj
, 
11071                                                const wxGridCellCoords
& topLeft
, 
11072                                                const wxGridCellCoords
& bottomRight
, 
11073                                                bool sel
, bool control
, 
11074                                                bool shift
, bool alt
, bool meta 
) 
11075         : wxNotifyEvent( type
, id 
) 
11077     m_topLeft 
= topLeft
; 
11078     m_bottomRight 
= bottomRight
; 
11080     m_control 
= control
; 
11085     SetEventObject(obj
); 
11089 IMPLEMENT_DYNAMIC_CLASS(wxGridEditorCreatedEvent
, wxCommandEvent
) 
11091 wxGridEditorCreatedEvent::wxGridEditorCreatedEvent(int id
, wxEventType type
, 
11092                                                    wxObject
* obj
, int row
, 
11093                                                    int col
, wxControl
* ctrl
) 
11094     : wxCommandEvent(type
, id
) 
11096     SetEventObject(obj
); 
11102 #endif // wxUSE_GRID