28#include <wx/tokenzr.h>
29#include <wx/clipbrd.h>
36#define COL_SEP wxT( '\t' )
37#define ROW_SEP wxT( '\n' )
38#define ROW_SEP_R wxT( '\r' )
43 m_addHandler( []( wxCommandEvent& ) {} )
51 m_addHandler( aAddHandler )
64 m_grid->Connect( wxEVT_GRID_CELL_LEFT_CLICK,
66 m_grid->Connect( wxEVT_GRID_CELL_LEFT_DCLICK,
68 m_grid->Connect( wxEVT_GRID_CELL_RIGHT_CLICK,
70 m_grid->Connect( wxEVT_GRID_LABEL_RIGHT_CLICK,
72 m_grid->Connect( wxEVT_GRID_LABEL_LEFT_CLICK,
76 m_grid->Connect( wxEVT_CHAR_HOOK,
78 m_grid->Connect( wxEVT_KEY_DOWN,
80 m_grid->Connect( wxEVT_UPDATE_UI,
84 m_grid->GetGridWindow()->Connect( wxEVT_MOTION,
92 wxGridCellEditor*
editor =
m_grid->GetCellEditor( aRow, aCol );
93 bool retval = (
dynamic_cast<wxTextEntry*
>(
editor ) );
102 wxGridCellRenderer* renderer =
m_grid->GetCellRenderer( aRow, aCol );
103 bool retval = (
dynamic_cast<wxGridCellBoolRenderer*
>( renderer ) );
112 return !
m_grid->IsEditable() ||
m_grid->IsReadOnly( aRow, aCol );
120 if( !aPreserveSelection )
123 m_grid->SetGridCursor( aRow, aCol );
126 wxGridTableBase* model =
m_grid->GetTable();
128 if( model->CanGetValueAs( aRow, aCol, wxGRID_VALUE_BOOL )
129 && model->CanSetValueAs( aRow, aCol, wxGRID_VALUE_BOOL ) )
131 model->SetValueAsBool( aRow, aCol, !model->GetValueAsBool( aRow, aCol ) );
135 if( model->GetValue( aRow, aCol ) == wxT(
"1" ) )
136 model->SetValue( aRow, aCol, wxT(
"0" ) );
138 model->SetValue( aRow, aCol, wxT(
"1" ) );
145 wxGridEvent event(
m_grid->GetId(), wxEVT_GRID_CELL_CHANGED,
m_grid, aRow, aCol );
146 event.SetString( model->GetValue( aRow, aCol ) );
147 m_grid->GetEventHandler()->ProcessEvent( event );
158 if(
m_grid->GetGridCursorRow() != aRow ||
m_grid->GetGridCursorCol() != aCol )
159 m_grid->SetGridCursor( aRow, aCol );
170 if(
m_grid->GetSelectionMode() == wxGrid::wxGridSelectRows )
172 wxArrayInt rows =
m_grid->GetSelectedRows();
174 if( rows.size() != 1 || rows.Item( 0 ) != aRow )
175 m_grid->SelectRow( aRow );
198 int row = aEvent.GetRow();
199 int col = aEvent.GetCol();
202 if( !aEvent.GetModifiers() )
204 bool toggled =
false;
227 wxString newVal =
m_grid->GetCellValue( row, col );
231 if( affectedRow == row )
234 m_grid->SetCellValue( affectedRow, col, newVal );
255 wxPoint pt = aEvent.GetPosition();
256 wxPoint pos =
m_grid->CalcScrolledPosition( wxPoint( pt.x, pt.y ) );
258 int col =
m_grid->XToCol( pos.x );
259 int row =
m_grid->YToRow( pos.y );
262 if( ( col == wxNOT_FOUND ) || ( row == wxNOT_FOUND ) || !
m_tooltipEnabled[col] )
264 m_grid->GetGridWindow()->SetToolTip( wxS(
"" ) );
269 m_grid->GetGridWindow()->SetToolTip(
m_grid->GetCellValue( row, col ) );
282 wxGridCellCoordsArray topLeft =
m_grid->GetSelectionBlockTopLeft();
283 wxGridCellCoordsArray botRight =
m_grid->GetSelectionBlockBottomRight();
285 wxArrayInt cols =
m_grid->GetSelectedCols();
286 wxArrayInt rows =
m_grid->GetSelectedRows();
288 if( topLeft.Count() && botRight.Count() )
296 else if( cols.Count() )
303 else if( rows.Count() )
340 for(
int i = 0; i <
m_grid->GetNumberCols(); ++i )
343 menu.AppendCheckItem(
id,
m_grid->GetColLabelValue( i ) );
344 menu.Check(
id,
m_grid->IsColShown( i ) );
347 m_grid->PopupMenu( &menu );
354 _(
"Clear selected cells placing original contents on clipboard" ) );
356 _(
"Copy selected cells to clipboard" ) );
358 _(
"Paste clipboard cells to matrix at current cell" ) );
360 _(
"Clear contents of selected cells" ) );
362 _(
"Select all cells" ) );
370 auto anyCellsWritable =
385 if( anyCellsWritable() )
395 if( wxTheClipboard->Open() )
397 if( wxTheClipboard->IsSupported( wxDF_TEXT )
398 || wxTheClipboard->IsSupported( wxDF_UNICODETEXT ) )
403 wxTheClipboard->Close();
406 m_grid->PopupMenu( &menu );
418 int menu_id =
event.GetId();
450 if(
m_grid->IsColShown( col ) )
461 bool handled =
false;
463 if( ( ev.GetKeyCode() == WXK_RETURN || ev.GetKeyCode() == WXK_NUMPAD_ENTER )
464 && ev.GetModifiers() == wxMOD_NONE
465 &&
m_grid->GetGridCursorRow() ==
m_grid->GetNumberRows() - 1 )
467 if(
m_grid->IsCellEditControlShown() )
474 wxCommandEvent
dummy;
479 else if( ev.GetModifiers() == wxMOD_CONTROL && ev.GetKeyCode() ==
'V' )
481 if(
m_grid->IsCellEditControlShown() && wxTheClipboard->Open() )
483 if( wxTheClipboard->IsSupported( wxDF_TEXT )
484 || wxTheClipboard->IsSupported( wxDF_UNICODETEXT ) )
486 wxTextDataObject data;
487 wxTheClipboard->GetData( data );
489 if( data.GetText().Contains(
COL_SEP ) || data.GetText().Contains(
ROW_SEP ) )
491 wxString stripped( data.GetText() );
492 stripped.Replace(
ROW_SEP,
" " );
494 stripped.Replace(
COL_SEP,
" " );
500 wxTheClipboard->Close();
504 else if( ev.GetKeyCode() == WXK_ESCAPE )
506 if(
m_grid->IsCellEditControlShown() )
520 if( ev.GetModifiers() == wxMOD_CONTROL && ev.GetKeyCode() ==
'A' )
525 else if( ev.GetModifiers() == wxMOD_CONTROL && ev.GetKeyCode() ==
'C' )
531 else if( ev.GetModifiers() == wxMOD_CONTROL && ev.GetKeyCode() ==
'V' )
537 else if( ev.GetModifiers() == wxMOD_CONTROL && ev.GetKeyCode() ==
'X' )
543 else if( !ev.GetModifiers() && ev.GetKeyCode() == WXK_DELETE )
551 if(
m_grid->IsEditable() && ev.GetKeyCode() ==
' ' )
556 if(
m_grid->GetSelectionMode() == wxGrid::wxGridSelectRows )
558 wxArrayInt rowSel =
m_grid->GetSelectedRows();
560 for(
unsigned int rowInd = 0; rowInd < rowSel.GetCount(); rowInd++ )
561 retVal |=
toggleCell( rowSel[rowInd], 0,
true );
565 else if(
m_grid->GetSelectionMode() == wxGrid::wxGridSelectColumns )
567 wxArrayInt colSel =
m_grid->GetSelectedCols();
569 for(
unsigned int colInd = 0; colInd < colSel.GetCount(); colInd++ )
570 retVal |=
toggleCell( 0, colSel[colInd],
true );
574 else if(
m_grid->GetSelectionMode() == wxGrid::wxGridSelectCells )
576 wxArrayInt rowSel =
m_grid->GetSelectedRows();
577 wxArrayInt colSel =
m_grid->GetSelectedCols();
578 wxGridCellCoordsArray cellSel =
m_grid->GetSelectedCells();
579 wxGridCellCoordsArray topLeft =
m_grid->GetSelectionBlockTopLeft();
580 wxGridCellCoordsArray botRight =
m_grid->GetSelectionBlockBottomRight();
583 for(
unsigned int cellInd = 0; cellInd < cellSel.GetCount(); cellInd++ )
585 retVal |=
toggleCell( cellSel[cellInd].GetRow(), cellSel[cellInd].GetCol(),
true );
589 for(
unsigned int colInd = 0; colInd < colSel.GetCount(); colInd++ )
591 for(
int row = 0; row <
m_grid->GetNumberRows(); row++ )
592 retVal |=
toggleCell( row, colSel[colInd],
true );
596 for(
unsigned int rowInd = 0; rowInd < rowSel.GetCount(); rowInd++ )
598 for(
int col = 0; col <
m_grid->GetNumberCols(); col++ )
599 retVal |=
toggleCell( rowSel[rowInd], col,
true );
603 for(
unsigned int blockInd = 0; blockInd < topLeft.GetCount(); blockInd++ )
605 wxGridCellCoords start = topLeft[blockInd];
606 wxGridCellCoords end = botRight[blockInd];
608 for(
int row = start.GetRow(); row <= end.GetRow(); row++ )
610 for(
int col = start.GetCol(); col <= end.GetCol(); col++ )
623 bool ctrl = ev.RawControlDown();
625 bool ctrl = ev.ControlDown();
628 if( ctrl && ev.GetKeyCode() == WXK_TAB )
635 while(
test && !
test->IsTopLevel() )
639 if(
test->HasFocus() )
642 if( !
test->GetChildren().empty() )
646 else if(
test->GetNextSibling() )
660 else if(
test &&
test->GetNextSibling() )
680 if(
m_grid->IsEditable() && wxTheClipboard->Open() )
682 if( wxTheClipboard->IsSupported( wxDF_TEXT )
683 || wxTheClipboard->IsSupported( wxDF_UNICODETEXT ) )
685 wxTextDataObject data;
687 wxTheClipboard->GetData( data );
692 wxTheClipboard->Close();
700 wxGridTableBase* tbl =
m_grid->GetTable();
702 const int cur_row =
m_grid->GetGridCursorRow();
703 const int cur_col =
m_grid->GetGridCursorCol();
708 bool is_selection =
false;
710 if( cur_row < 0 || cur_col < 0 )
716 if(
m_grid->GetSelectionMode() == wxGrid::wxGridSelectRows )
726 wxStringTokenizer rows( cb_text,
ROW_SEP, wxTOKEN_RET_EMPTY );
742 end_row = cur_row + rows.CountTokens();
744 if( end_row > tbl->GetNumberRows() )
748 for(
int ii = end_row - tbl->GetNumberRows(); ii > 0; --ii )
750 wxCommandEvent
dummy;
755 end_row = tbl->GetNumberRows();
762 for(
int row = start_row; row < end_row; ++row )
766 if( !rows.HasMoreTokens() )
767 rows.SetString( cb_text,
ROW_SEP, wxTOKEN_RET_EMPTY );
769 wxString rowTxt = rows.GetNextToken();
771 wxStringTokenizer cols( rowTxt,
COL_SEP, wxTOKEN_RET_EMPTY );
774 end_col = cur_col + cols.CountTokens();
776 for(
int col = start_col; col < end_col && col < tbl->GetNumberCols(); ++col )
779 if( !
m_grid->IsColShown( col ) )
787 if( !cols.HasMoreTokens() )
788 cols.SetString( rowTxt,
COL_SEP, wxTOKEN_RET_EMPTY );
790 wxString cellTxt = cols.GetNextToken();
794 if( tbl->CanSetValueAs( row, col, wxGRID_VALUE_STRING ) && !
isReadOnly( row, col ) )
796 tbl->SetValue( row, col, cellTxt );
798 wxGridEvent evt(
m_grid->GetId(), wxEVT_GRID_CELL_CHANGED,
m_grid, row, col );
799 m_grid->GetEventHandler()->ProcessEvent( evt );
810 if( doCopy && !wxTheClipboard->Open() )
813 wxGridTableBase* tbl =
m_grid->GetTable();
821 if( !
m_grid->IsColShown( col ) )
824 txt += tbl->GetValue( row, col );
834 tbl->SetValue( row, col, wxEmptyString );
843 wxTheClipboard->SetData(
new wxTextDataObject( txt ) );
844 wxTheClipboard->Flush();
845 wxTheClipboard->Close();
857 if(
m_grid->GetSelectionMode() == wxGrid::wxGridSelectRows )
859 int cursorRow =
m_grid->GetGridCursorRow();
860 bool cursorInSelectedRow =
false;
862 for(
int row :
m_grid->GetSelectedRows() )
864 if( row == cursorRow )
866 cursorInSelectedRow =
true;
871 if( !cursorInSelectedRow && cursorRow >= 0 )
872 m_grid->SelectRow( cursorRow );
bool isReadOnly(int aRow, int aCol)
void onGridMotion(wxMouseEvent &event)
void onGridLabelLeftClick(wxGridEvent &event)
virtual void paste_text(const wxString &cb_text)
void init()
Shared initialization for various ctors.
void getSelectedArea()
Puts the selected area into a sensible rectangle of m_sel_{row,col}_{start,count} above.
GRID_TRICKS(WX_GRID *aGrid)
void onKeyDown(wxKeyEvent &event)
void onPopupSelection(wxCommandEvent &event)
virtual void cutcopy(bool doCopy, bool doDelete)
virtual void doPopupSelection(wxCommandEvent &event)
bool isCheckbox(int aRow, int aCol)
std::function< void(wxCommandEvent &)> m_addHandler
virtual void showPopupMenu(wxMenu &menu, wxGridEvent &aEvent)
std::bitset< GRIDTRICKS_MAX_COL > m_tooltipEnabled
void onUpdateUI(wxUpdateUIEvent &event)
void onGridCellRightClick(wxGridEvent &event)
WX_GRID * m_grid
I don't own the grid, but he owns me.
bool isTextEntry(int aRow, int aCol)
void onGridCellLeftDClick(wxGridEvent &event)
void onGridCellLeftClick(wxGridEvent &event)
virtual bool handleDoubleClick(wxGridEvent &aEvent)
void onCharHook(wxKeyEvent &event)
virtual void paste_clipboard()
bool showEditor(int aRow, int aCol)
void onGridLabelRightClick(wxGridEvent &event)
bool toggleCell(int aRow, int aCol, bool aPreserveSelection=false)
bool CancelPendingChanges()
void ShowEditorOnMouseUp()
WxWidgets has a bunch of bugs in its handling of wxGrid mouse events which close cell editors right a...
bool CommitPendingChanges(bool aQuietMode=false)
Close any open cell edit controls.
@ GRIDTRICKS_FIRST_SHOWHIDE
std::vector< FAB_LAYER_COLOR > dummy