2018-05-30 11:52:19 +01:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2025-01-01 13:30:11 -08:00
|
|
|
* Copyright The KiCad Developers, see AUTHORS.txt for contributors.
|
2018-05-30 11:52:19 +01:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 3
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef KICAD_WX_GRID_H
|
|
|
|
#define KICAD_WX_GRID_H
|
|
|
|
|
2024-07-26 20:49:29 +00:00
|
|
|
#include <bitset>
|
2022-10-11 16:47:29 -07:00
|
|
|
#include <memory>
|
2024-07-26 20:49:29 +00:00
|
|
|
#include <utility>
|
2022-10-12 00:37:19 +01:00
|
|
|
#include <vector>
|
2024-07-26 20:49:29 +00:00
|
|
|
|
2022-03-21 16:44:28 -07:00
|
|
|
#include <wx/event.h>
|
2018-05-30 11:52:19 +01:00
|
|
|
#include <wx/grid.h>
|
2022-03-21 16:44:28 -07:00
|
|
|
#include <wx/version.h>
|
2024-07-26 20:49:29 +00:00
|
|
|
|
2022-10-11 18:01:47 +01:00
|
|
|
#include <libeval/numeric_evaluator.h>
|
2024-07-26 20:49:29 +00:00
|
|
|
#include <units_provider.h>
|
2022-03-21 16:44:28 -07:00
|
|
|
|
2024-04-13 23:25:57 +03:00
|
|
|
class wxTextEntryBase;
|
|
|
|
|
2018-05-30 11:52:19 +01:00
|
|
|
|
2024-05-16 15:50:57 +01:00
|
|
|
class WX_GRID_TABLE_BASE : public wxGridTableBase
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
wxGridCellAttr* enhanceAttr( wxGridCellAttr* aInputAttr, int aRow, int aCol,
|
|
|
|
wxGridCellAttr::wxAttrKind aKind );
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2018-05-30 11:52:19 +01:00
|
|
|
class WX_GRID : public wxGrid
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Constructor has to be wxFormBuilder-compatible
|
|
|
|
WX_GRID( wxWindow *parent, wxWindowID id,
|
|
|
|
const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxDefaultSize,
|
2018-08-19 17:10:14 +01:00
|
|
|
long style = wxWANTS_CHARS, const wxString& name = wxGridNameStr );
|
|
|
|
|
|
|
|
~WX_GRID() override;
|
2018-05-30 11:52:19 +01:00
|
|
|
|
2019-02-07 21:48:41 +00:00
|
|
|
/**
|
|
|
|
* Hide wxGrid's SetColLabelSize() method with one which makes sure the size is tall
|
|
|
|
* enough for the system GUI font.
|
2025-01-04 09:21:11 -05:00
|
|
|
*
|
2019-02-07 21:48:41 +00:00
|
|
|
* @param height
|
|
|
|
*/
|
2023-01-18 11:50:47 +00:00
|
|
|
void SetColLabelSize( int aHeight ); // Yes, we're hiding a non-virtual method
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hide wxGrid's SetLabelFont() because for some reason on MSW it's a one-shot and
|
|
|
|
* subsequent calls to it have no effect.
|
|
|
|
*/
|
|
|
|
void SetLabelFont( const wxFont& aFont ); // Yes, we're hiding a non-virtual method
|
2019-02-07 21:48:41 +00:00
|
|
|
|
2024-02-13 18:59:13 +00:00
|
|
|
/**
|
|
|
|
* Enable alternate row highlighting, where every odd row has a different background
|
2025-01-04 09:21:11 -05:00
|
|
|
* color than the even rows.
|
2024-02-13 18:59:13 +00:00
|
|
|
*
|
2025-01-04 09:21:11 -05:00
|
|
|
* @param aEnable flag to specify to enable alternate row striping in the grid.
|
2024-02-13 18:59:13 +00:00
|
|
|
*/
|
|
|
|
void EnableAlternateRowColors( bool aEnable = true );
|
|
|
|
|
2018-05-30 11:52:19 +01:00
|
|
|
/**
|
|
|
|
* Get a tokenized string containing the shown column indexes.
|
2025-01-04 09:21:11 -05:00
|
|
|
*
|
2018-05-30 11:52:19 +01:00
|
|
|
* Tokens are separated by spaces.
|
|
|
|
*/
|
2023-04-13 13:45:22 +01:00
|
|
|
wxString GetShownColumnsAsString();
|
|
|
|
std::bitset<64> GetShownColumns();
|
2018-05-30 11:52:19 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Show/hide the grid columns based on a tokenized string of shown column indexes.
|
|
|
|
*/
|
|
|
|
void ShowHideColumns( const wxString& shownColumns );
|
|
|
|
|
2023-04-13 13:45:22 +01:00
|
|
|
/**
|
|
|
|
* A more performant version of ShowHideColumns (primarily for OnUpdateUI handlers).
|
|
|
|
*/
|
|
|
|
void ShowHideColumns( const std::bitset<64>& aShownColumns );
|
|
|
|
|
2018-05-30 11:52:19 +01:00
|
|
|
/**
|
|
|
|
* Hide wxGrid's SetTable() method with one which doesn't mess up the grid column
|
|
|
|
* widths when setting the table.
|
|
|
|
*/
|
2018-08-19 17:10:14 +01:00
|
|
|
void SetTable( wxGridTableBase* table, bool aTakeOwnership = false );
|
2018-05-30 11:52:19 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Work-around for a bug in wxGrid which crashes when deleting the table if the
|
|
|
|
* cell edit control was not closed.
|
|
|
|
*/
|
|
|
|
void DestroyTable( wxGridTableBase* aTable );
|
|
|
|
|
2018-08-13 18:00:08 +01:00
|
|
|
/**
|
|
|
|
* Close any open cell edit controls.
|
2025-01-04 09:21:11 -05:00
|
|
|
*
|
|
|
|
* @param aQuietMode if true don't send events (ie: for row/col delete operations).
|
|
|
|
* @return false if validation failed.
|
2018-08-13 18:00:08 +01:00
|
|
|
*/
|
|
|
|
bool CommitPendingChanges( bool aQuietMode = false );
|
2023-04-10 15:12:35 +01:00
|
|
|
bool CancelPendingChanges();
|
2018-08-13 18:00:08 +01:00
|
|
|
|
2025-07-24 17:55:00 +01:00
|
|
|
/**
|
|
|
|
* Handles a row deletion event. This is a bit tricky due to the potential for stale
|
|
|
|
* selections, so we code it only once here.
|
|
|
|
*/
|
|
|
|
void OnDeleteRows( const std::function<void( int row )>& aDeleter );
|
|
|
|
|
|
|
|
void OnDeleteRows( const std::function<bool( int row )>& aFilter,
|
|
|
|
const std::function<void( int row )>& aDeleter );
|
|
|
|
|
2022-10-11 18:01:47 +01:00
|
|
|
/**
|
2025-01-04 09:21:11 -05:00
|
|
|
* Set a EUNITS_PROVIDER to enable use of unit- and eval-based Getters.
|
|
|
|
*
|
2022-10-11 18:01:47 +01:00
|
|
|
* @param aProvider
|
|
|
|
*/
|
2022-11-04 12:20:53 +00:00
|
|
|
void SetUnitsProvider( UNITS_PROVIDER* aProvider, int aCol = 0 );
|
2022-10-11 18:01:47 +01:00
|
|
|
|
|
|
|
void SetAutoEvalCols( const std::vector<int>& aCols ) { m_autoEvalCols = aCols; }
|
|
|
|
|
2025-02-10 20:57:51 +00:00
|
|
|
/**
|
|
|
|
* Set the unit and unit data type to use for a given column
|
|
|
|
*/
|
|
|
|
void SetAutoEvalColUnits( int col, EDA_UNITS aUnit, EDA_DATA_TYPE aUnitType );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the unit to use for a given column. The unit data type is inferred from the unit type
|
|
|
|
*/
|
|
|
|
void SetAutoEvalColUnits( int col, EDA_UNITS aUnit );
|
|
|
|
|
2022-10-11 18:01:47 +01:00
|
|
|
/**
|
|
|
|
* Apply standard KiCad unit and eval services to a numeric cell.
|
2025-01-04 09:21:11 -05:00
|
|
|
*
|
|
|
|
* @param aRow the cell row index to fetch.
|
|
|
|
* @param aCol the cell column index to fetch.
|
|
|
|
* @param aIsOptional if true, indicates to the unit provider the value is optional.
|
|
|
|
* @return the value held by the cell in internal units.
|
2022-10-11 18:01:47 +01:00
|
|
|
*/
|
|
|
|
int GetUnitValue( int aRow, int aCol );
|
|
|
|
|
2024-07-26 20:49:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply standard KiCad unit and eval services to a numeric cell.
|
2025-01-04 09:21:11 -05:00
|
|
|
*
|
|
|
|
* @param aRow the cell row index to fetch.
|
|
|
|
* @param aCol the cell column index to fetch.
|
|
|
|
* @return the value held by the cell in internal units.
|
2024-07-26 20:49:29 +00:00
|
|
|
*/
|
|
|
|
std::optional<int> GetOptionalUnitValue( int aRow, int aCol );
|
|
|
|
|
2022-10-11 18:01:47 +01:00
|
|
|
/**
|
|
|
|
* Set a unitized cell's value.
|
|
|
|
*/
|
|
|
|
void SetUnitValue( int aRow, int aCol, int aValue );
|
|
|
|
|
2024-07-26 20:49:29 +00:00
|
|
|
/**
|
2025-01-04 09:21:11 -05:00
|
|
|
* Set a unitized cell's optional value.
|
2024-07-26 20:49:29 +00:00
|
|
|
*/
|
|
|
|
void SetOptionalUnitValue( int aRow, int aCol, std::optional<int> aValue );
|
|
|
|
|
2019-02-09 15:31:31 -08:00
|
|
|
/**
|
2025-01-04 09:21:11 -05:00
|
|
|
* Calculate the specified column based on the actual size of the text on screen.
|
|
|
|
*
|
|
|
|
* @param aCol is the index of the column to resize. Specify -1 for the row labels.
|
|
|
|
* @param aHeader is the header in the width calculation.
|
|
|
|
* @param aContents is the full contents of the column.
|
|
|
|
* @param aKeep is the current size as a minimum value.
|
|
|
|
* @return The maximum value of all calculated widths.
|
2019-02-09 15:31:31 -08:00
|
|
|
*/
|
2022-08-18 17:56:36 +01:00
|
|
|
int GetVisibleWidth( int aCol, bool aHeader = true, bool aContents = true, bool aKeep = false );
|
2019-02-09 15:31:31 -08:00
|
|
|
|
2019-02-18 20:34:44 +01:00
|
|
|
/**
|
|
|
|
* Ensure the height of the row displaying the column labels is enough, even
|
2025-01-04 09:21:11 -05:00
|
|
|
* if labels are multiline texts.
|
2019-02-18 20:34:44 +01:00
|
|
|
*/
|
|
|
|
void EnsureColLabelsVisible();
|
|
|
|
|
2019-03-04 11:02:12 +00:00
|
|
|
/**
|
|
|
|
* WxWidgets has a bunch of bugs in its handling of wxGrid mouse events which close cell
|
2025-01-04 09:21:11 -05:00
|
|
|
* editors right after opening them.
|
|
|
|
*
|
|
|
|
* Helpfully, it already has a bunch of work-arounds in place (such as the SetInSetFocus()
|
|
|
|
* hack), including one to make slow clicks work. We re-purpose this hack to work-around
|
|
|
|
* the bugs when we want to open an editor.
|
2019-03-04 11:02:12 +00:00
|
|
|
*/
|
|
|
|
void ShowEditorOnMouseUp() { m_waitForSlowClick = true; }
|
2024-12-31 17:47:59 +00:00
|
|
|
void CancelShowEditorOnMouseUp() { m_waitForSlowClick = false; }
|
2019-03-04 11:02:12 +00:00
|
|
|
|
2021-08-10 12:59:53 +01:00
|
|
|
/**
|
|
|
|
* wxWidgets recently added an ASSERT which fires if the position is greater than or equal
|
|
|
|
* to the number of rows (even if the delete count is 0). Needless to say, this makes using
|
|
|
|
* DeleteRows for clearing a lot more cumbersome so we add a helper here.
|
|
|
|
*/
|
|
|
|
void ClearRows()
|
|
|
|
{
|
|
|
|
if( GetNumberRows() )
|
|
|
|
DeleteRows( 0, GetNumberRows() );
|
|
|
|
}
|
|
|
|
|
2024-04-13 23:25:57 +03:00
|
|
|
/**
|
|
|
|
* A helper function to set OS-specific margins for text-based cell editors.
|
|
|
|
*/
|
|
|
|
static void CellEditorSetMargins( wxTextEntryBase* aEntry );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A helper function to tweak sizes of text-based cell editors depending on OS.
|
|
|
|
*/
|
|
|
|
static void CellEditorTransformSizeRect( wxRect& aRect );
|
|
|
|
|
2024-12-31 13:07:26 +00:00
|
|
|
/**
|
|
|
|
* Grids that have column sizes automatically set to fill the available width don't want
|
|
|
|
* to shrink afterwards (because wxGrid reports the aggregate column size as the bestSize.
|
2025-01-04 09:21:11 -05:00
|
|
|
*
|
2024-12-31 13:07:26 +00:00
|
|
|
* @param aSize
|
|
|
|
*/
|
|
|
|
void OverrideMinSize( double aXPct, double aYPct )
|
|
|
|
{
|
|
|
|
wxSize size = DoGetBestSize();
|
|
|
|
m_minSizeOverride = wxSize( KiROUND( size.x * aXPct ), KiROUND( size.y * aYPct ) );
|
|
|
|
}
|
2025-01-04 09:21:11 -05:00
|
|
|
|
2024-12-31 13:07:26 +00:00
|
|
|
wxSize DoGetBestSize() const override
|
|
|
|
{
|
|
|
|
if( m_minSizeOverride )
|
|
|
|
return m_minSizeOverride.value();
|
|
|
|
else
|
|
|
|
return wxGrid::DoGetBestSize();
|
|
|
|
}
|
|
|
|
|
2018-05-30 11:52:19 +01:00
|
|
|
protected:
|
2021-07-15 15:26:35 -04:00
|
|
|
/**
|
2023-01-16 02:46:09 +00:00
|
|
|
* A re-implementation of wxGrid::DrawColLabel which left-aligns the first column and draws
|
|
|
|
* flat borders.
|
2021-07-15 15:26:35 -04:00
|
|
|
*/
|
2018-05-30 11:52:19 +01:00
|
|
|
void DrawColLabel( wxDC& dc, int col ) override;
|
2018-08-19 17:10:14 +01:00
|
|
|
|
2023-01-16 02:46:09 +00:00
|
|
|
/**
|
|
|
|
* A re-implementation of wxGrid::DrawRowLabel which draws flat borders.
|
|
|
|
*/
|
|
|
|
void DrawRowLabel( wxDC& dc, int row ) override;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A re-implementation of wxGrid::DrawCornerLabel which draws flat borders.
|
|
|
|
*/
|
|
|
|
void DrawCornerLabel( wxDC& dc ) override;
|
|
|
|
|
2018-10-09 21:27:15 +01:00
|
|
|
void onGridColMove( wxGridEvent& aEvent );
|
2022-02-04 12:36:27 +01:00
|
|
|
void onGridCellSelect( wxGridEvent& aEvent );
|
2022-10-11 18:01:47 +01:00
|
|
|
void onCellEditorShown( wxGridEvent& aEvent );
|
|
|
|
void onCellEditorHidden( wxGridEvent& aEvent );
|
|
|
|
|
2022-03-21 16:44:28 -07:00
|
|
|
void onDPIChanged(wxDPIChangedEvent& event);
|
|
|
|
|
2025-04-18 18:41:43 +01:00
|
|
|
UNITS_PROVIDER* getUnitsProvider( int aCol ) const
|
|
|
|
{
|
|
|
|
if( m_unitsProviders.contains( aCol ) )
|
|
|
|
return m_unitsProviders.at( aCol );
|
|
|
|
else
|
|
|
|
return m_unitsProviders.begin()->second;
|
|
|
|
}
|
|
|
|
|
2025-02-10 20:57:51 +00:00
|
|
|
/**
|
|
|
|
* Returns the units and data type associated with a given column
|
|
|
|
*/
|
|
|
|
std::pair<EDA_UNITS, EDA_DATA_TYPE> getColumnUnits( int aCol ) const;
|
|
|
|
|
2022-10-11 18:01:47 +01:00
|
|
|
protected:
|
|
|
|
bool m_weOwnTable;
|
|
|
|
|
2022-11-04 12:20:53 +00:00
|
|
|
std::map<int, UNITS_PROVIDER*> m_unitsProviders;
|
2022-10-11 18:01:47 +01:00
|
|
|
std::unique_ptr<NUMERIC_EVALUATOR> m_eval;
|
|
|
|
std::vector<int> m_autoEvalCols;
|
2025-02-10 20:57:51 +00:00
|
|
|
std::unordered_map<int, std::pair<EDA_UNITS, EDA_DATA_TYPE>> m_autoEvalColsUnits;
|
2022-10-11 18:01:47 +01:00
|
|
|
|
|
|
|
std::map< std::pair<int, int>, std::pair<wxString, wxString> > m_evalBeforeAfter;
|
2024-12-31 13:07:26 +00:00
|
|
|
|
|
|
|
std::optional<wxSize> m_minSizeOverride;
|
2018-05-30 11:52:19 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif //KICAD_WX_GRID_H
|