1 /////////////////////////////////////////////////////////////////////////// 
   2 // Name:        generic/gridsel.cpp 
   3 // Purpose:     wxGridSelection 
   8 // Copyright:   (c) Stefan Neis (Stefan.Neis@t-online.de) 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21     #pragma implementation "gridsel.h" 
  24 // For compilers that support precompilation, includes "wx/wx.h". 
  25 #include "wx/wxprec.h" 
  35 #include "wx/generic/gridsel.h" 
  37 // Some explanation for the members of the class: 
  38 // m_cellSelection stores individual selected cells 
  39 //   -- this is only used if m_selectionMode == wxGridSelectCells 
  40 // m_blockSelectionTopLeft and m_blockSelectionBottomRight 
  41 //   store the upper left and lower right corner of selected Blocks 
  42 // m_rowSelection and m_colSelection store individual selected 
  43 //   rows and columns; maybe those are superfluous and should be 
  46 wxGridSelection::wxGridSelection( wxGrid 
* grid
, 
  47                                   wxGrid::wxGridSelectionModes sel 
) 
  50     m_selectionMode 
= sel
; 
  53 bool wxGridSelection::IsSelection() 
  55   return ( m_cellSelection
.GetCount() || m_blockSelectionTopLeft
.GetCount() || 
  56            m_rowSelection
.GetCount() || m_colSelection
.GetCount() ); 
  59 bool wxGridSelection::IsInSelection ( int row
, int col 
) 
  63     // First check whether the given cell is individually selected 
  64     // (if m_selectionMode is wxGridSelectCells). 
  65     if ( m_selectionMode 
== wxGrid::wxGridSelectCells 
) 
  67         count 
= m_cellSelection
.GetCount(); 
  68         for ( size_t n 
= 0; n 
< count
; n
++ ) 
  70             wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
  71             if ( row 
== coords
.GetRow() && col 
== coords
.GetCol() ) 
  76     // Now check whether the given cell is 
  77     // contained in one of the selected blocks. 
  78     count 
= m_blockSelectionTopLeft
.GetCount(); 
  79     for ( size_t n 
= 0; n 
< count
; n
++ ) 
  81         wxGridCellCoords
& coords1 
= m_blockSelectionTopLeft
[n
]; 
  82         wxGridCellCoords
& coords2 
= m_blockSelectionBottomRight
[n
]; 
  83         if ( BlockContainsCell(coords1
.GetRow(), coords1
.GetCol(), 
  84                                coords2
.GetRow(), coords2
.GetCol(), 
  89     // Now check whether the given cell is 
  90     // contained in one of the selected rows 
  91     // (unless we are in column selection mode). 
  92     if ( m_selectionMode 
!= wxGrid::wxGridSelectColumns 
) 
  94         size_t count 
= m_rowSelection
.GetCount(); 
  95         for ( size_t n 
= 0; n 
< count
; n
++ ) 
  97             if ( row 
== m_rowSelection
[n
] ) 
 102     // Now check whether the given cell is 
 103     // contained in one of the selected columns 
 104     // (unless we are in row selection mode). 
 105     if ( m_selectionMode 
!= wxGrid::wxGridSelectRows 
) 
 107         size_t count 
= m_colSelection
.GetCount(); 
 108         for ( size_t n 
= 0; n 
< count
; n
++ ) 
 110             if ( col 
== m_colSelection
[n
] ) 
 117 // Change the selection mode 
 118 void wxGridSelection::SetSelectionMode(wxGrid::wxGridSelectionModes selmode
) 
 120     // if selection mode is unchanged return immediately 
 121     if (selmode 
== m_selectionMode
) 
 124     if ( m_selectionMode 
!= wxGrid::wxGridSelectCells 
) 
 126         // if changing form row to column selection 
 127         // or vice versa, clear the selection. 
 128         if ( selmode 
!= wxGrid::wxGridSelectCells 
) 
 131         m_selectionMode 
= selmode
; 
 135         // if changing from cell selection to something else, 
 136         // promote selected cells/blocks to whole rows/columns. 
 138         while ( ( n 
= m_cellSelection
.GetCount() ) > 0 ) 
 141             wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
 142             int row 
= coords
.GetRow(); 
 143             int col 
= coords
.GetCol(); 
 144             m_cellSelection
.RemoveAt(n
); 
 145             if (selmode 
== wxGrid::wxGridSelectRows
) 
 147             else // selmode == wxGridSelectColumns) 
 151         for (n 
= 0; n 
< m_blockSelectionTopLeft
.GetCount(); n
++) 
 152         // Note that m_blockSelectionTopLeft's size may be changing! 
 154             wxGridCellCoords
& coords 
= m_blockSelectionTopLeft
[n
]; 
 155             int topRow 
= coords
.GetRow(); 
 156             int leftCol 
= coords
.GetCol(); 
 157             coords 
= m_blockSelectionBottomRight
[n
]; 
 158             int bottomRow 
= coords
.GetRow(); 
 159             int rightCol 
= coords
.GetCol(); 
 160             if (selmode 
== wxGrid::wxGridSelectRows
) 
 162                 if (leftCol 
!= 0 || rightCol 
!= m_grid
->GetNumberCols() - 1 ) 
 164                     m_blockSelectionTopLeft
.RemoveAt(n
); 
 165                     m_blockSelectionBottomRight
.RemoveAt(n
); 
 166                     SelectBlock( topRow
, 0, 
 167                                  bottomRow
, m_grid
->GetNumberCols() - 1, 
 168                                  FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 171             else // selmode == wxGridSelectColumns) 
 173                 if (topRow 
!= 0 || bottomRow 
!= m_grid
->GetNumberRows() - 1 ) 
 175                     m_blockSelectionTopLeft
.RemoveAt(n
); 
 176                     m_blockSelectionBottomRight
.RemoveAt(n
); 
 177                     SelectBlock( 0, leftCol
, 
 178                                  m_grid
->GetNumberRows() - 1, rightCol
, 
 179                                  FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 183         m_selectionMode 
= selmode
; 
 187 void wxGridSelection::SelectRow( int row
, 
 188                                  bool ControlDown
,  bool ShiftDown
, 
 189                                  bool AltDown
, bool MetaDown 
) 
 191     if ( m_selectionMode 
== wxGrid::wxGridSelectColumns 
) 
 195     // Remove single cells contained in newly selected block. 
 196     if ( m_selectionMode 
== wxGrid::wxGridSelectCells 
) 
 198         count 
= m_cellSelection
.GetCount(); 
 199         for ( n 
= 0; n 
< count
; n
++ ) 
 201             wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
 202             if ( BlockContainsCell( row
, 0, row
, m_grid
->GetNumberCols() - 1, 
 203                                     coords
.GetRow(), coords
.GetCol() ) ) 
 205                 m_cellSelection
.RemoveAt(n
); 
 211     // Simplify list of selected blocks (if possible) 
 212     count 
= m_blockSelectionTopLeft
.GetCount(); 
 214     for ( n 
= 0; n 
< count
; n
++ ) 
 216         wxGridCellCoords
& coords1 
= m_blockSelectionTopLeft
[n
]; 
 217         wxGridCellCoords
& coords2 
= m_blockSelectionBottomRight
[n
]; 
 219         // Remove block if it is a subset of the row 
 220         if ( coords1
.GetRow() == row 
&& row 
== coords2
.GetRow() ) 
 222             m_blockSelectionTopLeft
.RemoveAt(n
); 
 223             m_blockSelectionBottomRight
.RemoveAt(n
); 
 226         else if ( coords1
.GetCol() == 0  && 
 227                   coords2
.GetCol() == m_grid
->GetNumberCols() - 1 ) 
 229             // silently return, if row is contained in block 
 230             if ( coords1
.GetRow() <= row 
&& row 
<= coords2
.GetRow() ) 
 232             // expand block, if it touched row 
 233             else if ( coords1
.GetRow() == row 
+ 1) 
 238             else if ( coords2
.GetRow() == row 
- 1) 
 246     // Unless we successfully handled the row, 
 247     // check whether row is already selected. 
 250         count 
= m_rowSelection
.GetCount(); 
 251         for ( n 
= 0; n 
< count
; n
++ ) 
 253             if ( row 
== m_rowSelection
[n
] ) 
 257         // Add row to selection 
 258         m_rowSelection
.Add(row
); 
 262     wxRect r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( row
, 0 ), 
 263                                           wxGridCellCoords( row
, m_grid
->GetNumberCols() - 1 ) ); 
 264     if ( !m_grid
->GetBatchCount() ) 
 265         ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 268     wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 269                                     wxEVT_GRID_RANGE_SELECT
, 
 271                                     wxGridCellCoords( row
, 0 ), 
 272                                     wxGridCellCoords( row
, m_grid
->GetNumberCols() - 1 ), 
 274                                     ControlDown
,  ShiftDown
, 
 277     m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 280 void wxGridSelection::SelectCol( int col
, 
 281                                  bool ControlDown
,  bool ShiftDown
, 
 282                                  bool AltDown
, bool MetaDown 
) 
 284     if ( m_selectionMode 
== wxGrid::wxGridSelectRows 
) 
 288     // Remove single cells contained in newly selected block. 
 289     if ( m_selectionMode 
== wxGrid::wxGridSelectCells 
) 
 291         count 
= m_cellSelection
.GetCount(); 
 292         for ( n 
= 0; n 
< count
; n
++ ) 
 294             wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
 295             if ( BlockContainsCell( 0, col
, m_grid
->GetNumberRows() - 1, col
, 
 296                                     coords
.GetRow(), coords
.GetCol() ) ) 
 298                 m_cellSelection
.RemoveAt(n
); 
 304     // Simplify list of selected blocks (if possible) 
 305     count 
= m_blockSelectionTopLeft
.GetCount(); 
 307     for ( n 
= 0; n 
< count
; n
++ ) 
 309         wxGridCellCoords
& coords1 
= m_blockSelectionTopLeft
[n
]; 
 310         wxGridCellCoords
& coords2 
= m_blockSelectionBottomRight
[n
]; 
 312         // Remove block if it is a subset of the column 
 313         if ( coords1
.GetCol() == col 
&& col 
== coords2
.GetCol() ) 
 315             m_blockSelectionTopLeft
.RemoveAt(n
); 
 316             m_blockSelectionBottomRight
.RemoveAt(n
); 
 319         else if ( coords1
.GetRow() == 0  && 
 320                   coords2
.GetRow() == m_grid
->GetNumberRows() - 1 ) 
 322             // silently return, if row is contained in block 
 323             if ( coords1
.GetCol() <= col 
&& col 
<= coords2
.GetCol() ) 
 325             // expand block, if it touched col 
 326             else if ( coords1
.GetCol() == col 
+ 1) 
 331             else if ( coords2
.GetCol() == col 
- 1) 
 339     // Unless we successfully handled the column, 
 340     // Check whether col is already selected. 
 343         count 
= m_colSelection
.GetCount(); 
 344         for ( n 
= 0; n 
< count
; n
++ ) 
 346             if ( col 
== m_colSelection
[n
] ) 
 350         // Add col to selection 
 351         m_colSelection
.Add(col
); 
 355     wxRect r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( 0, col 
), 
 356                                           wxGridCellCoords( m_grid
->GetNumberRows() - 1, col 
) ); 
 357     if ( !m_grid
->GetBatchCount() ) 
 358         ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 361     wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 362                                     wxEVT_GRID_RANGE_SELECT
, 
 364                                     wxGridCellCoords( 0, col 
), 
 365                                     wxGridCellCoords( m_grid
->GetNumberRows() - 1, col 
), 
 367                                     ControlDown
,  ShiftDown
, 
 370     m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 373 void wxGridSelection::SelectBlock( int topRow
, int leftCol
, 
 374                                    int bottomRow
, int rightCol
, 
 375                                    bool ControlDown
, bool ShiftDown
, 
 376                                    bool AltDown
, bool MetaDown
,  
 379     // Fix the coordinates of the block if needed. 
 380     if ( m_selectionMode 
== wxGrid::wxGridSelectRows 
) 
 383         rightCol 
= m_grid
->GetNumberCols() - 1; 
 385     else if ( m_selectionMode 
== wxGrid::wxGridSelectColumns 
) 
 388         bottomRow 
= m_grid
->GetNumberRows() - 1; 
 390     if ( topRow 
> bottomRow 
) 
 397     if ( leftCol 
> rightCol 
) 
 404     // Handle single cell selection in SelectCell. 
 405     // (MB: added check for selection mode here to prevent 
 406     //  crashes if, for example, we are select rows and the 
 407     //  grid only has 1 col) 
 408     if ( m_selectionMode 
== wxGrid::wxGridSelectCells 
&& 
 409          topRow 
== bottomRow 
&& leftCol 
== rightCol 
) 
 410         SelectCell( topRow
, leftCol
, ControlDown
,  ShiftDown
, 
 411                     AltDown
, MetaDown
, sendEvent 
); 
 414     // Remove single cells contained in newly selected block. 
 415     if ( m_selectionMode 
== wxGrid::wxGridSelectCells 
) 
 417         count 
= m_cellSelection
.GetCount(); 
 418         for ( n 
= 0; n 
< count
; n
++ ) 
 420             wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
 421             if ( BlockContainsCell( topRow
, leftCol
, bottomRow
, rightCol
, 
 422                                     coords
.GetRow(), coords
.GetCol() ) ) 
 424                 m_cellSelection
.RemoveAt(n
); 
 430     // If a block containing the selection is already selected, return, 
 431     // if a block contained in the selection is found, remove it. 
 433     count 
= m_blockSelectionTopLeft
.GetCount(); 
 434     for ( n 
= 0; n 
< count
; n
++ ) 
 436         wxGridCellCoords
& coords1 
= m_blockSelectionTopLeft
[n
]; 
 437         wxGridCellCoords
& coords2 
= m_blockSelectionBottomRight
[n
]; 
 438         switch ( BlockContain( coords1
.GetRow(), coords1
.GetCol(), 
 439                                coords2
.GetRow(), coords2
.GetCol(), 
 440                                topRow
, leftCol
, bottomRow
, rightCol 
) ) 
 445             m_blockSelectionTopLeft
.RemoveAt(n
); 
 446             m_blockSelectionBottomRight
.RemoveAt(n
); 
 453     // If a row containing the selection is already selected, return, 
 454     // if a row contained in newly selected block is found, remove it. 
 455     if ( m_selectionMode 
!= wxGrid::wxGridSelectColumns 
) 
 457         count 
= m_rowSelection
.GetCount(); 
 458         for ( n 
= 0; n 
< count
; n
++ ) 
 460             switch ( BlockContain( m_rowSelection
[n
], 0, 
 461                                    m_rowSelection
[n
], m_grid
->GetNumberCols()-1, 
 462                                    topRow
, leftCol
, bottomRow
, rightCol 
) ) 
 467                 m_rowSelection
.RemoveAt(n
); 
 474     if ( m_selectionMode 
!= wxGrid::wxGridSelectRows 
) 
 476         count 
= m_colSelection
.GetCount(); 
 477         for ( n 
= 0; n 
< count
; n
++ ) 
 479             switch ( BlockContain( 0, m_colSelection
[n
], 
 480                                    m_grid
->GetNumberRows()-1, m_colSelection
[n
],  
 481                                    topRow
, leftCol
, bottomRow
, rightCol 
) ) 
 486                 m_colSelection
.RemoveAt(n
); 
 493     m_blockSelectionTopLeft
.Add( wxGridCellCoords( topRow
, leftCol 
) ); 
 494     m_blockSelectionBottomRight
.Add( wxGridCellCoords( bottomRow
, rightCol 
) ); 
 497     wxRect r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( topRow
, leftCol 
), 
 498                                           wxGridCellCoords( bottomRow
, rightCol 
) ); 
 499     if ( !m_grid
->GetBatchCount() ) 
 500         ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 502     // Send Event, if not disabled. 
 505         wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 506                                         wxEVT_GRID_RANGE_SELECT
, 
 508                                         wxGridCellCoords( topRow
, leftCol 
), 
 509                                         wxGridCellCoords( bottomRow
, rightCol 
), 
 511                                         ControlDown
, ShiftDown
, 
 513         m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 517 void wxGridSelection::SelectCell( int row
, int col
, 
 518                                   bool ControlDown
, bool ShiftDown
, 
 519                                   bool AltDown
, bool MetaDown
, 
 522     if ( m_selectionMode 
== wxGrid::wxGridSelectRows 
) 
 524         SelectBlock(row
, 0, row
, m_grid
->GetNumberCols() - 1, 
 525                     ControlDown
, ShiftDown
, AltDown
, MetaDown
, sendEvent
); 
 528     else if ( m_selectionMode 
== wxGrid::wxGridSelectColumns 
) 
 530         SelectBlock(0, col
, m_grid
->GetNumberRows() - 1, col
, 
 531                     ControlDown
, ShiftDown
, AltDown
, MetaDown
, sendEvent
); 
 534     else if ( IsInSelection ( row
, col 
) ) 
 536     m_cellSelection
.Add( wxGridCellCoords( row
, col 
) ); 
 539     wxRect r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( row
, col 
), 
 540                                           wxGridCellCoords( row
, col 
) ); 
 541     if ( !m_grid
->GetBatchCount() ) 
 542         ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 547         wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 548                                         wxEVT_GRID_RANGE_SELECT
, 
 550                                         wxGridCellCoords( row
, col 
), 
 551                                         wxGridCellCoords( row
, col 
), 
 553                                         ControlDown
, ShiftDown
, 
 555         m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 559 void wxGridSelection::ToggleCellSelection( int row
, int col
, 
 560                                            bool ControlDown
, bool ShiftDown
, 
 561                                            bool AltDown
, bool MetaDown 
) 
 563     // if the cell is not selected, select it 
 564     if ( !IsInSelection ( row
, col 
) ) 
 566         SelectCell( row
, col
, ControlDown
, ShiftDown
, 
 571     // otherwise deselect it. This can be simple or more or 
 572     // less difficult, depending on how the cell is selected. 
 575     // The simplest case: The cell is contained in m_cellSelection 
 576     // Then it can't be contained in rows/cols/block (since those 
 577     // would remove the cell from m_cellSelection on creation), so 
 578     // we just have to remove it from m_cellSelection. 
 580     if ( m_selectionMode 
== wxGrid::wxGridSelectCells 
) 
 582         count 
= m_cellSelection
.GetCount(); 
 583         for ( n 
= 0; n 
< count
; n
++ ) 
 585             wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
 586             if ( row 
== coords
.GetRow() && col 
== coords
.GetCol() ) 
 589                 r 
= m_grid
->BlockToDeviceRect( m_cellSelection
[n
], 
 590                                                m_cellSelection
[n
] ); 
 591                 m_cellSelection
.RemoveAt(n
); 
 593                 if ( !m_grid
->GetBatchCount() ) 
 594                     ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 597                 wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 598                                                 wxEVT_GRID_RANGE_SELECT
, 
 600                                                 wxGridCellCoords( row
, col 
), 
 601                                                 wxGridCellCoords( row
, col 
), 
 603                                                 ControlDown
, ShiftDown
, 
 605                 m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 611     // The most difficult case: The cell is member of one or even several 
 612     // blocks. Split each such block in up to 4 new parts, that don't 
 613     // contain the cell to be selected, like this: 
 614     // |---------------------------| 
 618     // |---------------------------| 
 619     // |   part 3   |x|   part 4   | 
 620     // |---------------------------| 
 624     // |---------------------------| 
 625     //   (The x marks the newly deselected cell). 
 626     // Note: in row selection mode, we only need part1 and part2; 
 627     //       in column selection mode, we only need part 3 and part4, 
 628     //          which are expanded to whole columns automatically! 
 630     count 
= m_blockSelectionTopLeft
.GetCount(); 
 631     for ( n 
= 0; n 
< count
; n
++ ) 
 633         wxGridCellCoords
& coords1 
= m_blockSelectionTopLeft
[n
]; 
 634         wxGridCellCoords
& coords2 
= m_blockSelectionBottomRight
[n
]; 
 635         int topRow 
= coords1
.GetRow(); 
 636         int leftCol 
= coords1
.GetCol(); 
 637         int bottomRow 
= coords2
.GetRow(); 
 638         int rightCol 
= coords2
.GetCol(); 
 639         if ( BlockContainsCell( topRow
, leftCol
, bottomRow
, rightCol
, 
 643             m_blockSelectionTopLeft
.RemoveAt(n
); 
 644             m_blockSelectionBottomRight
.RemoveAt(n
); 
 646             // add up to 4 smaller blocks and set update region 
 647             if ( m_selectionMode 
!= wxGrid::wxGridSelectColumns 
) 
 650                     SelectBlock( topRow
, leftCol
, row 
- 1, rightCol
, 
 651                                  FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 652                 if ( bottomRow 
> row 
) 
 653                     SelectBlock( row 
+ 1, leftCol
, bottomRow
, rightCol
, 
 654                                  FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 656             if ( m_selectionMode 
!= wxGrid::wxGridSelectRows 
) 
 659                     SelectBlock( row
, leftCol
, row
, col 
- 1, 
 660                                  FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 661                 if ( rightCol 
> col 
) 
 662                     SelectBlock( row
, col 
+ 1, row
, rightCol
, 
 663                                  FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 668     // remove a cell from a row, adding up to two new blocks 
 669     if ( m_selectionMode 
!= wxGrid::wxGridSelectColumns 
) 
 671         count 
= m_rowSelection
.GetCount(); 
 672         for ( n 
= 0; n 
< count
; n
++ ) 
 674             if ( m_rowSelection
[n
] == row 
) 
 676                 m_rowSelection
.RemoveAt(n
); 
 678                 if (m_selectionMode 
== wxGrid::wxGridSelectCells
) 
 681                         SelectBlock( row
, 0, row
, col 
- 1, 
 682                                      FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 683                     if ( col 
< m_grid
->GetNumberCols() - 1 ) 
 684                         SelectBlock( row
, col 
+ 1, 
 685                                      row
, m_grid
->GetNumberCols() - 1, 
 686                                      FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 692     // remove a cell from a column, adding up to two new blocks 
 693     if ( m_selectionMode 
!= wxGrid::wxGridSelectRows 
) 
 695         count 
= m_colSelection
.GetCount(); 
 696         for ( n 
= 0; n 
< count
; n
++ ) 
 698             if ( m_colSelection
[n
] == col 
) 
 700                 m_colSelection
.RemoveAt(n
); 
 702                 if (m_selectionMode 
== wxGrid::wxGridSelectCells
) 
 705                         SelectBlock( 0, col
, row 
- 1, col
, 
 706                                      FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 707                     if ( row 
< m_grid
->GetNumberRows() - 1 ) 
 708                         SelectBlock( row 
+ 1, col
, 
 709                                      m_grid
->GetNumberRows() - 1, col
, 
 710                                      FALSE
, FALSE
, FALSE
, FALSE
, FALSE 
); 
 716     // Refresh the screen and send the event; according to m_selectionMode, 
 717     // we need to either update only the cell, or the whole row/column. 
 719     switch (m_selectionMode
) 
 721       case wxGrid::wxGridSelectCells
: 
 723           r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( row
, col 
), 
 724                                          wxGridCellCoords( row
, col 
) ); 
 725           if ( !m_grid
->GetBatchCount() ) 
 726               ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 727           wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 728                                           wxEVT_GRID_RANGE_SELECT
, 
 730                                           wxGridCellCoords( row
, col 
), 
 731                                           wxGridCellCoords( row
, col 
), 
 733                                           ControlDown
, ShiftDown
, 
 735           m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 738       case wxGrid::wxGridSelectRows
: 
 740           r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( row
, 0 ), 
 741                                          wxGridCellCoords( row
, m_grid
->GetNumberCols() - 1 ) ); 
 742           if ( !m_grid
->GetBatchCount() ) 
 743               ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 744           wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 745                                           wxEVT_GRID_RANGE_SELECT
, 
 747                                           wxGridCellCoords( row
, 0 ), 
 748                                           wxGridCellCoords( row
, m_grid
->GetNumberCols() - 1 ), 
 750                                           ControlDown
, ShiftDown
, 
 752           m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 755       case wxGrid::wxGridSelectColumns
: 
 757           r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( 0, col 
), 
 758                                          wxGridCellCoords( m_grid
->GetNumberRows() - 1, col 
) ); 
 759           if ( !m_grid
->GetBatchCount() ) 
 760               ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 761           wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 762                                           wxEVT_GRID_RANGE_SELECT
, 
 764                                           wxGridCellCoords( 0, col 
), 
 765                                           wxGridCellCoords( m_grid
->GetNumberRows() - 1, col 
), 
 767                                           ControlDown
, ShiftDown
, 
 769           m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 775 void wxGridSelection::ClearSelection() 
 779     // deselect all invidiual cells and update the screen 
 780     if ( m_selectionMode 
== wxGrid::wxGridSelectCells 
) 
 783         while( ( n 
= m_cellSelection
.GetCount() ) > 0) 
 787             r 
= m_grid
->BlockToDeviceRect( m_cellSelection
[n
], 
 788                                            m_cellSelection
[n
] ); 
 789             m_cellSelection
.RemoveAt(n
); 
 790             if ( !m_grid
->GetBatchCount() ) 
 791                 ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
); 
 795     // deselect all blocks and update the screen 
 796     while( ( n 
= m_blockSelectionTopLeft
.GetCount() ) > 0) 
 800         r 
= m_grid
->BlockToDeviceRect( m_blockSelectionTopLeft
[n
], 
 801                                        m_blockSelectionBottomRight
[n
] ); 
 802         m_blockSelectionTopLeft
.RemoveAt(n
); 
 803         m_blockSelectionBottomRight
.RemoveAt(n
); 
 804         if ( !m_grid
->GetBatchCount() ) 
 805             ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
);    
 808     // deselect all rows and update the screen 
 809     if ( m_selectionMode 
!= wxGrid::wxGridSelectColumns 
) 
 811         while( ( n 
= m_rowSelection
.GetCount() ) > 0) 
 814             int & row 
= m_rowSelection
[n
]; 
 816             r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( row
, 0 ), 
 817                                            wxGridCellCoords( row
, m_grid
->GetNumberCols() - 1 ) ); 
 818             m_rowSelection
.RemoveAt(n
); 
 819             if ( !m_grid
->GetBatchCount() ) 
 820                 ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
);    
 824     // deselect all columns and update the screen 
 825     if ( m_selectionMode 
!= wxGrid::wxGridSelectRows 
) 
 827         while( ( n 
= m_colSelection
.GetCount() ) > 0) 
 830             int & col 
= m_colSelection
[n
]; 
 832             r 
= m_grid
->BlockToDeviceRect( wxGridCellCoords( 0, col 
), 
 833                                            wxGridCellCoords( m_grid
->GetNumberRows() - 1, col 
) ); 
 834             m_colSelection
.RemoveAt(n
); 
 835             if ( !m_grid
->GetBatchCount() ) 
 836                 ((wxWindow 
*)m_grid
->m_gridWin
)->Refresh( FALSE
, &r 
);    
 840     // One deselection event, indicating deselection of _all_ cells. 
 841     // (No finer grained events for each of the smaller regions 
 842     //  deselected above!) 
 843     wxGridRangeSelectEvent 
gridEvt( m_grid
->GetId(), 
 844                                     wxEVT_GRID_RANGE_SELECT
, 
 846                                     wxGridCellCoords( 0, 0 ), 
 847                                     wxGridCellCoords( m_grid
->GetNumberRows() - 1, 
 848                                                       m_grid
->GetNumberCols() - 1 ), 
 851     m_grid
->GetEventHandler()->ProcessEvent(gridEvt
); 
 855 void wxGridSelection::UpdateRows( size_t pos
, int numRows 
) 
 857     size_t count 
= m_cellSelection
.GetCount(); 
 859     for ( n 
= 0; n 
< count
; n
++ ) 
 861         wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
 862         wxCoord row 
= coords
.GetRow(); 
 863         if ((size_t)row 
>= pos
) 
 867                 // If rows inserted, increase row counter where necessary 
 868                 coords
.SetRow(row 
+ numRows
); 
 870             else if (numRows 
< 0) 
 872                 // If rows deleted ... 
 873                 if ((size_t)row 
>= pos 
- numRows
) 
 875                     // ...either decrement row counter (if row still exists)... 
 876                     coords
.SetRow(row 
+ numRows
); 
 880                     // ...or remove the attribute 
 881                     m_cellSelection
.RemoveAt(n
); 
 888     count 
= m_blockSelectionTopLeft
.GetCount(); 
 889     for ( n 
= 0; n 
< count
; n
++ ) 
 891         wxGridCellCoords
& coords1 
= m_blockSelectionTopLeft
[n
]; 
 892         wxGridCellCoords
& coords2 
= m_blockSelectionBottomRight
[n
]; 
 893         wxCoord row1 
= coords1
.GetRow(); 
 894         wxCoord row2 
= coords2
.GetRow(); 
 895         if ((size_t)row2 
>= pos
) 
 899                 // If rows inserted, increase row counter where necessary 
 900                 coords2
.SetRow(row2 
+ numRows
); 
 901                 if ( (size_t)row1 
>= pos 
) 
 902                     coords1
.SetRow(row1 
+ numRows
); 
 904             else if (numRows 
< 0) 
 906                 // If rows deleted ... 
 907                 if ((size_t)row2 
>= pos 
- numRows
) 
 909                     // ...either decrement row counter (if row still exists)... 
 910                     coords2
.SetRow(row2 
+ numRows
); 
 911                     if ( (size_t) row1 
>= pos
) 
 912                         coords1
.SetRow( wxMax(row1 
+ numRows
, (int) pos
) ); 
 917                     if ( (size_t) row1 
>= pos
) 
 919                         // ...or remove the attribute 
 920                         m_blockSelectionTopLeft
.RemoveAt(n
); 
 921                         m_blockSelectionBottomRight
.RemoveAt(n
); 
 931     count 
= m_rowSelection
.GetCount(); 
 932     for ( n 
= 0; n 
< count
; n
++ ) 
 934         int & rowOrCol 
= m_rowSelection
[n
]; 
 935         if ( (size_t)rowOrCol 
>= pos 
) 
 939                 // If rows inserted, include row counter where necessary 
 942             else if ( numRows 
< 0) 
 944                 // If rows deleted, either decrement row counter (if row still exists) 
 945                 if ((size_t)rowOrCol 
>= pos 
- numRows
) 
 949                     m_rowSelection
.RemoveAt(n
); 
 955     // No need to touch selected columns, unless we removed _all_ 
 956     // rows, in this case, we remove all columns from the selection. 
 957     if ( !m_grid
->GetNumberRows() ) 
 958         m_colSelection
.Clear(); 
 961 void wxGridSelection::UpdateCols( size_t pos
, int numCols 
) 
 963     size_t count 
= m_cellSelection
.GetCount(); 
 965     for ( n 
= 0; n 
< count
; n
++ ) 
 967         wxGridCellCoords
& coords 
= m_cellSelection
[n
]; 
 968         wxCoord col 
= coords
.GetCol(); 
 969         if ((size_t)col 
>= pos
) 
 973                 // If rows inserted, increase row counter where necessary 
 974                 coords
.SetCol(col 
+ numCols
); 
 976             else if (numCols 
< 0) 
 978                 // If rows deleted ... 
 979                 if ((size_t)col 
>= pos 
- numCols
) 
 981                     // ...either decrement row counter (if row still exists)... 
 982                     coords
.SetCol(col 
+ numCols
); 
 986                     // ...or remove the attribute 
 987                     m_cellSelection
.RemoveAt(n
); 
 994     count 
= m_blockSelectionTopLeft
.GetCount(); 
 995     for ( n 
= 0; n 
< count
; n
++ ) 
 997         wxGridCellCoords
& coords1 
= m_blockSelectionTopLeft
[n
]; 
 998         wxGridCellCoords
& coords2 
= m_blockSelectionBottomRight
[n
]; 
 999         wxCoord col1 
= coords1
.GetCol(); 
1000         wxCoord col2 
= coords2
.GetCol(); 
1001         if ((size_t)col2 
>= pos
) 
1005                 // If rows inserted, increase row counter where necessary 
1006                 coords2
.SetCol(col2 
+ numCols
); 
1007                 if ( (size_t)col1 
>= pos 
) 
1008                     coords1
.SetCol(col1 
+ numCols
); 
1010             else if (numCols 
< 0) 
1012                 // If cols deleted ... 
1013                 if ((size_t)col2 
>= pos 
- numCols
) 
1015                     // ...either decrement col counter (if col still exists)... 
1016                     coords2
.SetCol(col2 
+ numCols
); 
1017                     if ( (size_t) col1 
>= pos
) 
1018                         coords1
.SetCol( wxMax(col1 
+ numCols
, (int) pos
) ); 
1023                     if ( (size_t) col1 
>= pos
) 
1025                         // ...or remove the attribute 
1026                         m_blockSelectionTopLeft
.RemoveAt(n
); 
1027                         m_blockSelectionBottomRight
.RemoveAt(n
); 
1031                         coords2
.SetCol(pos
); 
1037     count 
= m_colSelection
.GetCount(); 
1038     for ( n 
= 0; n 
< count
; n
++ ) 
1040         int & rowOrCol 
= m_colSelection
[n
]; 
1041         if ( (size_t)rowOrCol 
>= pos 
) 
1045                 // If cols inserted, include col counter where necessary 
1046                 rowOrCol 
+= numCols
; 
1048             else if ( numCols 
< 0) 
1050                 // If cols deleted, either decrement col counter (if col still exists) 
1051                 if ((size_t)rowOrCol 
>= pos 
- numCols
) 
1052                     rowOrCol 
+= numCols
; 
1055                     m_colSelection
.RemoveAt(n
); 
1062     // No need to touch selected rows, unless we removed _all_ 
1063     // columns, in this case, we remove all rows from the selection. 
1064     if ( !m_grid
->GetNumberCols() ) 
1065         m_rowSelection
.Clear(); 
1068 int wxGridSelection::BlockContain( int topRow1
, int leftCol1
, 
1069                                    int bottomRow1
, int rightCol1
, 
1070                                    int topRow2
, int leftCol2
, 
1071                                    int bottomRow2
, int rightCol2 
) 
1072 // returns 1, if Block1 contains Block2, 
1073 //        -1, if Block2 contains Block1, 
1076     if ( topRow1 
<= topRow2 
&& bottomRow2 
<= bottomRow1 
&& 
1077          leftCol1 
<= leftCol2 
&& rightCol2 
<= rightCol1 
) 
1079     else if ( topRow2 
<= topRow1 
&& bottomRow1 
<= bottomRow2 
&& 
1080               leftCol2 
<= leftCol1 
&& rightCol1 
<= rightCol2 
)