26#include <wx/filedlg.h>
27#include <wx/wfstream.h>
29#include <wx/tokenzr.h>
56#define BOOL_TRUE _HKI( "True" )
57#define BOOL_FALSE _HKI( "False" )
59#define UNITS_ALL _HKI( "ALL" )
60#define DEMORGAN_ALL _HKI( "ALL" )
61#define DEMORGAN_STD _HKI( "Standard" )
62#define DEMORGAN_ALT _HKI( "Alternate" )
88 default: wxFAIL;
return wxEmptyString;
95 return wxGetTranslation( aNativeLabel ).IsSameAs( aStr, aCaseSensitive )
96 || aStr.IsSameAs( aNativeLabel, aCaseSensitive );
144 val << PinTypeNames()[static_cast<int>( aPin.
GetType() )];
147 val << PinShapeNames()[static_cast<int>( aPin.
GetShape() )];
183 case BODY_STYLE::BASE:
186 case BODY_STYLE::DEMORGAN:
195 wxFAIL_MSG( wxString::Format(
"Invalid field id %d", aFieldId ) );
220 if(
PinTypeNames().Index( aValue,
false ) != wxNOT_FOUND )
288 wxFAIL_MSG( wxString::Format(
"Invalid field id %d", aFieldId ) );
303 wxCHECK_MSG(
false, wxEmptyString,
"Invalid BOOL_FORMAT" );
308 if( aValue == wxS(
"1" ) )
312 else if( aValue == wxS(
"0" ) )
325 aReporter.
Report( wxString::Format(
_(
"The value '%s' can't be converted to boolean correctly, "
326 "it has been interpreted as 'False'" ),
341 wxGridCellCoordsArray topLeft = aGrid->GetSelectionBlockTopLeft();
342 wxGridCellCoordsArray botRight = aGrid->GetSelectionBlockBottomRight();
344 wxArrayInt cols = aGrid->GetSelectedCols();
345 wxArrayInt rows = aGrid->GetSelectedRows();
347 if( topLeft.Count() && botRight.Count() )
349 *aRowStart = topLeft[0].GetRow();
350 *aRowCount = botRight[0].GetRow() - *aRowStart + 1;
352 else if( cols.Count() )
355 *aRowCount = aGrid->GetNumberRows();
357 else if( rows.Count() )
359 *aRowStart = rows[0];
360 *aRowCount = rows.Count();
364 *aRowStart = aGrid->GetGridCursorRow();
365 *aRowCount = *aRowStart >= 0 ? 1 : 0;
376 const std::vector<SCH_PIN*>& aOrigSelectedPins ) :
399 GetView()->ForceRefresh();
423 wxGrid*
grid = GetView();
425 if(
grid->GetGridCursorRow() == aRow &&
grid->GetGridCursorCol() == aCol
426 &&
grid->IsCellEditControlShown() )
437 static wxString
GetValue(
const std::vector<SCH_PIN*>& pins,
int aCol,
463 if( fieldValue.length() )
464 fieldValue += wxT(
", " );
470 if( !fieldValue.Length() )
472 else if( val != fieldValue )
480 void SetValue(
int aRow,
int aCol,
const wxString &aValue )
override
485 wxString value = aValue;
496 if(
m_eval->Process( value ) )
508 std::vector<SCH_PIN*> pins =
m_rows[ aRow ];
514 wxStringTokenizer tokenizer( value,
"," );
517 while( tokenizer.HasMoreTokens() )
519 wxString pinName = tokenizer.GetNextToken();
522 pinName.Trim(
true ).Trim(
false );
524 if( i < pins.size() )
527 pins.at( i )->SetNumber( pinName );
559 pins.push_back( newPin );
566 while( pins.size() > i )
589 static int findRow(
const std::vector<std::vector<SCH_PIN*>>& aRowSet,
const wxString& aName )
591 for(
size_t i = 0; i < aRowSet.size(); ++i )
593 if( aRowSet[ i ][ 0 ] && aRowSet[ i ][ 0 ]->GetName() == aName )
600 static bool compare(
const std::vector<SCH_PIN*>& lhs,
const std::vector<SCH_PIN*>& rhs,
603 wxString lhStr =
GetValue( lhs, sortCol, parentFrame );
604 wxString rhStr =
GetValue( rhs, sortCol, parentFrame );
610 lhStr =
GetValue( lhs, sortCol, parentFrame );
611 rhStr =
GetValue( rhs, sortCol, parentFrame );
618 auto cmp = [ ascending ](
const auto a,
const auto b )
653 void RebuildRows(
const std::vector<SCH_PIN*>& aPins,
bool groupByName,
bool groupBySelection )
656 std::vector<SCH_PIN*> clear_flags;
658 clear_flags.reserve( aPins.size() );
662 if( groupBySelection )
665 pin->ClearTempFlags();
667 int firstSelectedRow;
668 int selectedRowCount;
672 for(
int ii = 0; ii < selectedRowCount; ++ii )
677 clear_flags.push_back(
pin );
684 grid->CommitPendingChanges(
true );
686 wxGridTableMessage msg(
this, wxGRIDTABLE_NOTIFY_ROWS_DELETED, 0,
m_rows.size() );
687 GetView()->ProcessTableMessage( msg );
692 if( groupBySelection )
693 m_rows.emplace_back( std::vector<SCH_PIN*>() );
695 std::set<wxString> selectedNumbers;
698 selectedNumbers.insert(
pin->GetNumber() );
701 const auto pinIsInEditorSelection = [&](
SCH_PIN*
pin )
705 if( selectedNumbers.count(
pin->GetNumber() ) == 0 )
715 if( selectedPin->GetNumber() ==
pin->GetNumber()
716 && selectedPin->GetName() ==
pin->GetName()
717 && selectedPin->GetUnit() ==
pin->GetUnit()
718 && selectedPin->GetBodyStyle() ==
pin->GetBodyStyle()
731 const bool includedByUnit =
733 const bool includedByBodyStyle =
737 if( includedByUnit && includedByBodyStyle && includedBySelection )
743 else if( groupBySelection && (
pin->GetFlags() &
CANDIDATE ) )
748 m_rows.emplace_back( std::vector<SCH_PIN*>() );
749 rowIndex =
m_rows.size() - 1;
757 bool ascending =
true;
759 if( GetView() && GetView()->GetSortingColumn() != wxNOT_FOUND )
761 sortCol = GetView()->GetSortingColumn();
762 ascending = GetView()->IsSortOrderAscending();
765 for( std::vector<SCH_PIN*>& row :
m_rows )
768 if( !groupBySelection )
773 wxGridTableMessage msg(
this, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, (
int)
m_rows.size() );
774 GetView()->ProcessTableMessage( msg );
776 if( groupBySelection )
777 GetView()->SelectRow( 0 );
787 [ aSortCol, ascending,
this ](
const std::vector<SCH_PIN*>& lhs,
788 const std::vector<SCH_PIN*>& rhs ) ->
bool
790 return compare( lhs, rhs, aSortCol, ascending, m_frame );
796 std::sort( aRow.begin(), aRow.end(),
799 return PIN_NUMBERS::Compare( lhs->GetNumber(), rhs->GetNumber() ) < 0;
805 std::vector<SCH_PIN*> row;
806 row.push_back( aPin );
811 wxGridTableMessage msg(
this, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, 1 );
812 GetView()->ProcessTableMessage( msg );
818 std::vector<SCH_PIN*> removedRow =
m_rows[ aRow ];
824 wxGridTableMessage msg(
this, wxGRIDTABLE_NOTIFY_ROWS_DELETED, aRow, 1 );
825 GetView()->ProcessTableMessage( msg );
847 std::vector<std::vector<SCH_PIN*>>
m_rows;
860 std::unique_ptr<NUMERIC_EVALUATOR>
m_eval;
873 void ExportData( std::vector<SCH_PIN*>& aPins,
const wxString& aToFile )
const
875 std::vector<int> exportCols {
891 std::vector<std::vector<wxString>> exportTable;
892 exportTable.reserve( aPins.size() + 1 );
894 std::vector<wxString> headers;
895 for(
int col : exportCols )
899 exportTable.emplace_back( std::move( headers ) );
906 std::vector<wxString>& cols = exportTable.emplace_back( 0 );
907 cols.reserve( exportCols.size() );
908 for(
int col : exportCols )
910 cols.emplace_back( formatter.
Format( *
pin, col ) );
914 if( !aToFile.IsEmpty() )
916 wxFileOutputStream os( aToFile );
951 std::vector<std::vector<wxString>> csvData;
954 if( !
path.IsEmpty() )
965 std::vector<std::unique_ptr<SCH_PIN>> pins;
975 for(
size_t i = 1; i < csvData.size(); ++i )
977 std::vector<wxString>& cols = csvData[i];
979 auto pin = std::make_unique<SCH_PIN>( &aSym );
982 size_t maxCol = std::min( headerCols.size(), cols.size() );
984 for(
size_t j = 0; j < maxCol; ++j )
990 formatter.
UpdatePin( *
pin, cols[j], headerCols[j], aSym );
993 pins.emplace_back( std::move(
pin ) );
1003 wxFD_OPEN | wxFD_FILE_MUST_EXIST );
1005 if( dlg.ShowModal() == wxID_CANCEL )
1006 return wxEmptyString;
1008 return dlg.GetPath();
1013 std::vector<COL_ORDER> colOrder;
1014 wxArrayString unknownHeaders;
1016 for(
size_t i = 0; i < aHeaderRow.size(); ++i )
1021 unknownHeaders.push_back( aHeaderRow[i] );
1023 colOrder.push_back( col );
1026 if( unknownHeaders.size() )
1028 wxString msg = wxString::Format(
_(
"Unknown columns in data: %s. These columns will be ignored." ),
1043 const std::vector<SCH_PIN*>& aSelectedPins ) :
1045 m_editFrame( parent ),
1059 m_grid->SetSelectionMode( wxGrid::wxGridSelectRows );
1069 wxGridCellAttr* attr;
1071 attr =
new wxGridCellAttr;
1072 attr->SetReadOnly(
true );
1075 attr =
new wxGridCellAttr;
1082 attr =
new wxGridCellAttr;
1089 attr =
new wxGridCellAttr;
1093 orientationNames ) );
1097 attr =
new wxGridCellAttr;
1098 wxArrayString unitNames;
1099 unitNames.push_back( wxGetTranslation(
UNITS_ALL ) );
1107 attr =
new wxGridCellAttr;
1108 wxArrayString demorganNames;
1109 demorganNames.push_back( wxGetTranslation(
DEMORGAN_ALL ) );
1110 demorganNames.push_back( wxGetTranslation(
DEMORGAN_STD ) );
1111 demorganNames.push_back( wxGetTranslation(
DEMORGAN_ALT ) );
1115 attr =
new wxGridCellAttr;
1116 attr->SetRenderer(
new wxGridCellBoolRenderer() );
1117 attr->SetEditor(
new wxGridCellBoolEditor() );
1118 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
1139 GetSizer()->SetSizeHints(
this);
1202 m_grid->PopEventHandler(
true );
1265 int sortCol = aEvent.GetCol();
1270 if(
m_grid->IsSortingBy( sortCol ) )
1272 ascending = !
m_grid->IsSortOrderAscending();
1284 [&]() -> std::pair<int, int>
1315 m_pins.push_back( newPin );
1372 if( candidate->GetNumber() == pins.at( 0 )->GetNumber() )
1401 m_grid->SetColLabelAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
1417 m_grid->SetColLabelAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
1428 if( event.IsChecked() )
1443 if( event.IsChecked() )
1490 std::vector<std::unique_ptr<SCH_PIN>> newPins = importer.
ImportData( fromFile, *
m_symbol );
1494 int ret = wxMessageBox( reporter.
GetMessages(),
_(
"Errors" ), wxOK | wxCANCEL | wxICON_ERROR,
this );
1497 if( ret == wxCANCEL )
1501 if( !newPins.size() )
1513 for(
auto& newPin : newPins )
1514 m_pins.push_back( newPin.release() );
1528 wxString filePath = wxEmptyString;
1535 wxFD_SAVE | wxFD_OVERWRITE_PROMPT );
1537 if( dlg.ShowModal() == wxID_CANCEL )
1540 filePath = dlg.GetPath();
1543 std::vector<SCH_PIN*> pinsToExport;
1550 pinsToExport.push_back(
pin );
1559 exporter.
ExportData( pinsToExport, filePath );
1568 wxGridUpdateLocker deferRepaintsTillLeavingScope;
1585 width -=
m_grid->GetColSize( i );
1597 wxSize new_size =
event.GetSize();
1619 if( !
m_grid->IsCellEditControlShown() )
1623 int firstSelectedRow;
1624 int selectedRowCount;
1644 int retval = wxID_CANCEL;
1663 EndDialog( retval );
1678 EndDialog( retval );
1688 if(
pin->GetNumber().Length() )
1693 m_pin_count->SetLabel( wxString::Format( wxT(
"%u" ), (
unsigned)
m_pins.size() ) );
constexpr EDA_IU_SCALE schIUScale
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
void WriteLines(const std::vector< std::vector< wxString > > &aRows)
Write a vector of rows to the stream.
Class DIALOG_LIB_EDIT_PIN_TABLE_BASE.
wxRadioButton * m_rbExportOnlyShownPins
STD_BITMAP_BUTTON * m_refreshButton
wxStaticText * m_pin_numbers_summary
BITMAP_BUTTON * m_divider1
wxButton * m_ButtonsCancel
STD_BITMAP_BUTTON * m_addButton
wxStaticText * m_pin_count
wxRadioButton * m_rbReplaceAll
wxStaticText * m_duplicate_pins
wxCheckBox * m_cbFilterSelected
wxChoice * m_bodyStyleFilter
wxCheckBox * m_cbFilterByBodyStyle
wxCheckBox * m_cbFilterByUnit
wxButton * m_btnExportToFile
STD_BITMAP_BUTTON * m_deleteButton
wxButton * m_groupSelected
wxButton * m_btnImportFromFile
bool m_modified
true when there are unsaved changes
void OnExportButtonClick(wxCommandEvent &event) override
void OnRebuildRows(wxCommandEvent &event) override
int m_originalColWidths[COL_COUNT]
void OnCancel(wxCommandEvent &event) override
void OnUpdateUI(wxUpdateUIEvent &event) override
void RemovePin(SCH_PIN *pin)
void OnColSort(wxGridEvent &aEvent)
void OnImportButtonClick(wxCommandEvent &event) override
void OnGroupSelected(wxCommandEvent &event) override
void OnClose(wxCloseEvent &event) override
void OnSize(wxSizeEvent &event) override
PIN_TABLE_DATA_MODEL * m_dataModel
DIALOG_LIB_EDIT_PIN_TABLE(SYMBOL_EDIT_FRAME *parent, LIB_SYMBOL *aSymbol, const std::vector< SCH_PIN * > &aSelectedPins)
~DIALOG_LIB_EDIT_PIN_TABLE() override
void AddPin(SCH_PIN *pin)
void OnCellSelected(wxGridEvent &event) override
bool TransferDataToWindow() override
void OnAddRow(wxCommandEvent &event) override
void OnDeleteRow(wxCommandEvent &event) override
SYMBOL_EDIT_FRAME * m_editFrame
std::bitset< 64 > m_columnsShown
void OnCellEdited(wxGridEvent &event) override
std::vector< SCH_PIN * > m_pins
bool TransferDataFromWindow() override
void OnFilterCheckBox(wxCommandEvent &event) override
void OnFilterChoice(wxCommandEvent &event) override
void SetupStandardButtons(std::map< int, wxString > aLabels={})
bool IsQuasiModal() const
void EndQuasiModal(int retCode)
The base frame for deriving all KiCad main window classes.
The base class for create windows for drawing purpose.
virtual void ClearFocus()
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
Add mouse and command handling (such as cut, copy, and paste) to a WX_GRID instance.
Define a library symbol object.
bool IsMulti() const override
std::vector< SCH_PIN * > GetPins(int aUnit, int aBodyStyle) const
Return a list of pin object pointers from the draw item list.
bool HasAlternateBodyStyle() const override
Test if symbol has more than one body conversion type (DeMorgan).
static wxString LetterSubReference(int aUnit, wxChar aInitialLetter)
wxString GetName() const override
void RemoveDrawItem(SCH_ITEM *aItem)
Remove draw aItem from list.
int GetUnitCount() const override
void AddDrawItem(SCH_ITEM *aItem, bool aSort=true)
Add a new draw aItem to the draw object list and sort according to aSort.
A singleton reporter that reports to nowhere.
wxString GetDuplicates() const
Gets a formatted string of all the pins that have duplicate numbers.
static int Compare(const wxString &lhs, const wxString &rhs)
void insert(value_type const &v)
wxString GetSummary() const
static bool compare(const std::vector< SCH_PIN * > &lhs, const std::vector< SCH_PIN * > &rhs, int sortCol, bool ascending, EDA_DRAW_FRAME *parentFrame)
void RebuildRows(const std::vector< SCH_PIN * > &aPins, bool groupByName, bool groupBySelection)
void SetUnitFilter(int aFilter)
static int findRow(const std::vector< std::vector< SCH_PIN * > > &aRowSet, const wxString &aName)
std::map< std::pair< std::vector< SCH_PIN * >, int >, wxString > m_evalOriginal
std::vector< std::vector< SCH_PIN * > > m_rows
int GetNumberCols() override
const std::vector< SCH_PIN * > & m_origSelectedPins
The pins in the symbol that are selected at dialog start.
bool IsEmptyCell(int row, int col) override
DIALOG_LIB_EDIT_PIN_TABLE * m_pinTable
std::vector< SCH_PIN * > GetRowPins(int aRow)
static wxString GetValue(const std::vector< SCH_PIN * > &pins, int aCol, EDA_DRAW_FRAME *aParentFrame)
PIN_TABLE_DATA_MODEL(SYMBOL_EDIT_FRAME *aFrame, DIALOG_LIB_EDIT_PIN_TABLE *aPinTable, LIB_SYMBOL *aSymbol, const std::vector< SCH_PIN * > &aOrigSelectedPins)
void AppendRow(SCH_PIN *aPin)
std::unique_ptr< NUMERIC_EVALUATOR > m_eval
void SetValue(int aRow, int aCol, const wxString &aValue) override
wxString GetValue(int aRow, int aCol) override
void SetFilterBySelection(bool aFilter)
void SortRows(int aSortCol, bool ascending)
SYMBOL_EDIT_FRAME * m_frame
int GetNumberRows() override
void onUnitsChanged(wxCommandEvent &aEvent)
void SetBodyStyleFilter(int aFilter)
wxString GetColLabelValue(int aCol) override
std::vector< SCH_PIN * > RemoveRow(int aRow)
void SortPins(std::vector< SCH_PIN * > &aRow)
void ExportData(std::vector< SCH_PIN * > &aPins, const wxString &aToFile) const
UNITS_PROVIDER & m_unitsProvider
PIN_TABLE_EXPORT(UNITS_PROVIDER &aUnitsProvider)
wxString promptForFile() const
std::vector< std::unique_ptr< SCH_PIN > > ImportData(bool aFromFile, LIB_SYMBOL &aSym) const
std::vector< COL_ORDER > getColOrderFromCSV(const std::vector< wxString > &aHeaderRow) const
PIN_TABLE_IMPORT(EDA_BASE_FRAME &aFrame, REPORTER &aReporter)
A pure virtual class used to derive REPORTER objects from.
virtual REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED)
Report a string with a given severity.
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
virtual void SetBodyStyle(int aBodyStyle)
virtual void SetUnit(int aUnit)
int GetNumberTextSize() const
void SetNumber(const wxString &aNumber)
void SetVisible(bool aVisible)
void ChangeLength(int aLength)
Change the length of a pin and adjust its position based on orientation.
void SetOrientation(PIN_ORIENTATION aOrientation)
void SetName(const wxString &aName)
void SetPosition(const VECTOR2I &aPos) override
const wxString & GetName() const
PIN_ORIENTATION GetOrientation() const
void SetNumberTextSize(int aSize)
void SetShape(GRAPHIC_PINSHAPE aShape)
VECTOR2I GetPosition() const override
int GetNameTextSize() const
void SetType(ELECTRICAL_PINTYPE aType)
const wxString & GetNumber() const
GRAPHIC_PINSHAPE GetShape() const
ELECTRICAL_PINTYPE GetType() const
void SetNameTextSize(int aSize)
The symbol library editor main window.
bool IsSymbolAlias() const
Return true if aLibId is an alias for the editor screen symbol.
bool GetShowDeMorgan() const
SYMBOL_EDITOR_SETTINGS * GetSettings() const
bool IsSymbolEditable() const
Test if a symbol is loaded and can be edited.
LIB_SYMBOL * GetCurSymbol() const
Return the current symbol being edited or NULL if none selected.
void FocusOnItem(EDA_ITEM *aItem) override
Focus on a particular canvas item.
EDA_UNITS GetUserUnits() const
wxString StringFromValue(double aValue, bool aAddUnitLabel=false, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
Converts aValue in internal units into a united string.
int ValueFromString(const wxString &aTextValue, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
Converts aTextValue in aUnits to internal units used by the frame.
void ShowHideColumns(const wxString &shownColumns)
Show/hide the grid columns based on a tokenized string of shown column indexes.
void SetTable(wxGridTableBase *table, bool aTakeOwnership=false)
Hide wxGrid's SetTable() method with one which doesn't mess up the grid column widths when setting th...
void DestroyTable(wxGridTableBase *aTable)
Work-around for a bug in wxGrid which crashes when deleting the table if the cell edit control was no...
void OnDeleteRows(const std::function< void(int row)> &aDeleter)
Handles a row deletion event.
wxString GetShownColumnsAsString()
Get a tokenized string containing the shown column indexes.
void OnAddRow(const std::function< std::pair< int, int >()> &aAdder)
std::bitset< 64 > GetShownColumns()
bool CommitPendingChanges(bool aQuietMode=false)
Close any open cell edit controls.
A wrapper for reporting to a wxString object.
bool HasMessage() const override
Returns true if the reporter client is non-empty.
const wxString & GetMessages() const
bool SaveTabularDataToClipboard(const std::vector< std::vector< wxString > > &aData)
Store tabular data to the system clipboard.
bool GetTabularDataFromClipboard(std::vector< std::vector< wxString > > &aData)
Attempt to get tabular data from the clipboard.
bool HandleUnsavedChanges(wxWindow *aParent, const wxString &aMessage, const std::function< bool()> &aSaveFunction)
Display a dialog with Save, Cancel and Discard Changes buttons.
This file is part of the common library.
bool AutoDecodeCSV(const wxString &aInput, std::vector< std::vector< wxString > > &aData)
Try to guess the format of a T/CSV file and decode it into aData.
static wxString GetPinTableColLabel(int aCol)
Get the label for a given column in the pin table.
static bool MatchTranslationOrNative(const wxString &aStr, const wxString &aNativeLabel, bool aCaseSensitive)
void getSelectedArea(WX_GRID *aGrid, int *aRowStart, int *aRowCount)
static COL_ORDER GetColTypeForString(const wxString &aStr)
#define CANDIDATE
flag indicating that the structure is connected
static std::map< int, wxString > shapeNames
static const std::string CsvFileExtension
static wxString CsvTsvFileWildcard()
static wxString CsvFileWildcard()
const std::vector< BITMAPS > & PinTypeIcons()
ELECTRICAL_PINTYPE
The symbol library pin object electrical types used in ERC tests.
const wxArrayString & PinTypeNames()
int PinOrientationIndex(PIN_ORIENTATION code)
const wxArrayString & PinShapeNames()
const std::vector< BITMAPS > & PinShapeIcons()
const wxArrayString & PinOrientationNames()
PIN_ORIENTATION
The symbol library pin object orientations.
const std::vector< BITMAPS > & PinOrientationIcons()
wxString SafeReadFile(const wxString &aFilePath, const wxString &aReadType)
Nominally opens a file and reads it into a string.
int StrNumCmp(const wxString &aString1, const wxString &aString2, bool aIgnoreCase)
Compare two strings with alphanumerical content.
void AccumulateDescriptions(wxString &aDesc, const T &aItemCollection)
Build a comma-separated list from a collection of wxStrings.
constexpr int MilsToIU(int mils) const
#define INDETERMINATE_STATE
Used for holding indeterminate values, such as with multiple selections holding different values or c...
VECTOR2< int32_t > VECTOR2I
Definition of file extensions used in Kicad.