summaryrefslogtreecommitdiff
path: root/pcbnew/dialogs/dialog_design_rules.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'pcbnew/dialogs/dialog_design_rules.cpp')
-rw-r--r--pcbnew/dialogs/dialog_design_rules.cpp1082
1 files changed, 1082 insertions, 0 deletions
diff --git a/pcbnew/dialogs/dialog_design_rules.cpp b/pcbnew/dialogs/dialog_design_rules.cpp
new file mode 100644
index 0000000..80fe0f8
--- /dev/null
+++ b/pcbnew/dialogs/dialog_design_rules.cpp
@@ -0,0 +1,1082 @@
+/**
+ * @file dialog_design_rules.cpp
+ */
+
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2004-2009 Jean-Pierre Charras, jp.charras at wanadoo.fr
+ * Copyright (C) 2009 Dick Hollenbeck, dick@softplc.com
+ * Copyright (C) 2009-2015 KiCad Developers, see change_log.txt for contributors.
+ *
+ * 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 2
+ * 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
+ */
+
+
+/* functions relative to the design rules editor
+ */
+#include <fctsys.h>
+#include <class_drawpanel.h>
+#include <base_units.h>
+#include <confirm.h>
+#include <pcbnew.h>
+#include <wxPcbStruct.h>
+#include <class_board_design_settings.h>
+
+#include <pcbnew_id.h>
+#include <class_track.h>
+#include <macros.h>
+#include <html_messagebox.h>
+
+#include <dialog_design_rules.h>
+#include <wx/generic/gridctrl.h>
+#include <dialog_design_rules_aux_helper_class.h>
+
+#include <boost/make_shared.hpp>
+
+// Column labels for net lists
+#define NET_TITLE _( "Net" )
+#define CLASS_TITLE _( "Class" )
+
+// Field Positions on rules grid
+enum {
+ GRID_CLEARANCE,
+ GRID_TRACKSIZE,
+ GRID_VIASIZE,
+ GRID_VIADRILL,
+ GRID_uVIASIZE,
+ GRID_uVIADRILL
+};
+
+const wxString DIALOG_DESIGN_RULES::wildCard = _( "* (Any)" );
+
+// dialog should remember its previously selected tab
+int DIALOG_DESIGN_RULES::s_LastTabSelection = -1;
+
+// methods for the helper class NETS_LIST_CTRL
+
+wxString NETS_LIST_CTRL::OnGetItemText( long item, long column ) const
+{
+ if( column == 0 )
+ {
+ if( item < (long) m_Netnames.GetCount() )
+ return m_Netnames[item];
+ else
+ return wxEmptyString;
+ }
+ else if( item < (long) m_Classnames.GetCount() )
+ return m_Classnames[item];
+
+ return wxEmptyString;
+}
+
+
+void NETS_LIST_CTRL::SetRowItems( unsigned aRow,
+ const wxString& aNetname,
+ const wxString& aNetclassName )
+{
+ // insert blanks if aRow is larger than existing row count
+ unsigned cnt = m_Netnames.GetCount();
+
+ if( cnt <= aRow )
+ m_Netnames.Add( wxEmptyString, aRow - cnt + 1 );
+
+ cnt = m_Classnames.GetCount();
+
+ if( cnt <= aRow )
+ m_Classnames.Add( wxEmptyString, aRow - cnt + 1 );
+
+ if( (int)aRow <= GetItemCount() )
+ SetItemCount( aRow + 1 );
+
+ m_Netnames[aRow] = aNetname;
+ m_Classnames[aRow] = aNetclassName;
+}
+
+
+/**
+ * Function EnsureGridColumnWidths
+ * resizes all the columns in a wxGrid based only on the requirements of the
+ * column titles and not on the grid cell requirements, assuming that the grid
+ * cell width requirements are narrower than the column title requirements.
+ */
+
+// @todo: maybe move this to common.cpp if it works.
+void EnsureGridColumnWidths( wxGrid* aGrid )
+{
+ wxScreenDC sDC;
+
+ sDC.SetFont( aGrid->GetLabelFont() );
+
+ int colCount = aGrid->GetNumberCols();
+
+ for( int col = 0; col<colCount; ++col )
+ {
+ // add two spaces to the text and size it.
+ wxString colText = aGrid->GetColLabelValue( col ) + wxT( " " );
+
+ wxSize needed = sDC.GetTextExtent( colText );
+
+ // set the width of this column
+ aGrid->SetColSize( col, needed.x );
+ }
+}
+
+
+DIALOG_DESIGN_RULES::DIALOG_DESIGN_RULES( PCB_EDIT_FRAME* parent ) :
+ DIALOG_DESIGN_RULES_BASE( parent )
+{
+ m_Parent = parent;
+ SetAutoLayout( true );
+
+ EnsureGridColumnWidths( m_grid ); // override any column widths set by wxformbuilder.
+
+ wxListItem column0;
+ wxListItem column1;
+
+ column0.Clear();
+ column1.Clear();
+
+ column0.SetMask( wxLIST_MASK_TEXT );
+ column1.SetMask( wxLIST_MASK_TEXT );
+
+ column0.SetText( NET_TITLE );
+ column1.SetText( CLASS_TITLE );
+
+ m_leftListCtrl->InsertColumn( 0, column0 );
+ m_leftListCtrl->InsertColumn( 1, column1 );
+ m_leftListCtrl->SetColumnWidth( 0, wxLIST_AUTOSIZE );
+ m_leftListCtrl->SetColumnWidth( 1, wxLIST_AUTOSIZE );
+
+ m_rightListCtrl->InsertColumn( 0, column0 );
+ m_rightListCtrl->InsertColumn( 1, column1 );
+ m_rightListCtrl->SetColumnWidth( 0, wxLIST_AUTOSIZE );
+ m_rightListCtrl->SetColumnWidth( 1, wxLIST_AUTOSIZE );
+
+ // if user has been into the dialog before, go back to same tab
+ if( s_LastTabSelection != -1 )
+ {
+ m_DRnotebook->SetSelection( s_LastTabSelection );
+ }
+
+ InitDialogRules();
+ m_sdbSizer1OK->SetDefault();
+
+ // Allow tabbing out of grid controls.
+ m_grid->SetTabBehaviour( wxGrid::Tab_Leave );
+ m_gridViaSizeList->SetTabBehaviour( wxGrid::Tab_Leave );
+ m_gridTrackWidthList->SetTabBehaviour( wxGrid::Tab_Leave );
+
+ Layout();
+
+ FixOSXCancelButtonIssue();
+
+ // Now all widgets have the size fixed, call FinishDialogSettings
+ FinishDialogSettings();
+}
+
+
+void DIALOG_DESIGN_RULES::InitDialogRules()
+{
+ // @todo: Move the initialization code into TransferDataToWindow() to follow wxWidgets
+ // dialog data transfer convention.
+ SetFocus();
+ SetReturnCode( 0 );
+
+ m_Pcb = m_Parent->GetBoard();
+ m_BrdSettings = &m_Pcb->GetDesignSettings();
+
+ // Initialize the Rules List
+ InitRulesList();
+
+ // copy all NETs into m_AllNets by adding them as NETCUPs.
+
+ // @todo go fix m_Pcb->SynchronizeNetsAndNetClasses() so that the netcode==0 is not
+ // present in the BOARD::m_NetClasses
+ NETCLASSES& netclasses = m_BrdSettings->m_NetClasses;
+ NETCLASSPTR netclass = netclasses.GetDefault();
+
+ // Initialize list of nets for Default Net Class
+ for( NETCLASS::iterator name = netclass->begin(); name != netclass->end(); ++name )
+ {
+ m_AllNets.push_back( NETCUP( *name, netclass->GetName() ) );
+ }
+
+ // Initialize list of nets for others (custom) Net Classes
+ for( NETCLASSES::const_iterator nc = netclasses.begin(); nc != netclasses.end(); ++nc )
+ {
+ netclass = nc->second;
+
+ for( NETCLASS::const_iterator name = netclass->begin(); name != netclass->end(); ++name )
+ {
+ m_AllNets.push_back( NETCUP( *name, netclass->GetName() ) );
+ }
+ }
+
+ InitializeRulesSelectionBoxes();
+ InitGlobalRules();
+}
+
+
+void DIALOG_DESIGN_RULES::InitGlobalRules()
+{
+ AddUnitSymbol( *m_ViaMinTitle );
+ AddUnitSymbol( *m_ViaMinDrillTitle );
+ AddUnitSymbol( *m_MicroViaMinSizeTitle );
+ AddUnitSymbol( *m_MicroViaMinDrillTitle );
+ AddUnitSymbol( *m_TrackMinWidthTitle );
+
+ PutValueInLocalUnits( *m_SetViasMinSizeCtrl, m_BrdSettings->m_ViasMinSize );
+ PutValueInLocalUnits( *m_SetViasMinDrillCtrl, m_BrdSettings->m_ViasMinDrill );
+
+ if( m_BrdSettings->m_BlindBuriedViaAllowed )
+ m_OptViaType->SetSelection( 1 );
+
+ m_AllowMicroViaCtrl->SetSelection( m_BrdSettings->m_MicroViasAllowed ? 1 : 0 );
+ PutValueInLocalUnits( *m_SetMicroViasMinSizeCtrl, m_BrdSettings->m_MicroViasMinSize );
+ PutValueInLocalUnits( *m_SetMicroViasMinDrillCtrl, m_BrdSettings->m_MicroViasMinDrill );
+ PutValueInLocalUnits( *m_SetTrackMinWidthCtrl, m_BrdSettings->m_TrackMinWidth );
+
+ // Initialize Vias and Tracks sizes lists.
+ // note we display only extra values, never the current netclass value.
+ // (the first value in history list)
+ m_TracksWidthList = m_BrdSettings->m_TrackWidthList;
+ m_TracksWidthList.erase( m_TracksWidthList.begin() ); // remove the netclass value
+ m_ViasDimensionsList = m_BrdSettings->m_ViasDimensionsList;
+ m_ViasDimensionsList.erase( m_ViasDimensionsList.begin() ); // remove the netclass value
+ InitDimensionsLists();
+}
+
+
+void DIALOG_DESIGN_RULES::InitDimensionsLists()
+{
+ wxString msg;
+
+ // Compute the column widths here, after setting texts
+ msg = wxT("000000.000000"); // This is a very long text to display values.
+ // Actual values are shorter.
+ m_gridViaSizeList->SetCellValue( 0, 0, msg );
+ m_gridViaSizeList->SetCellValue( 0, 1, msg );
+ m_gridTrackWidthList->SetCellValue( 0, 0, msg );
+ m_gridViaSizeList->SetColMinimalWidth( 0, 150 );
+ m_gridViaSizeList->SetColMinimalWidth( 1, 150 );
+ m_gridViaSizeList->AutoSizeColumns( true );
+ m_gridTrackWidthList->SetColMinimalWidth( 0, 150 );
+ m_gridTrackWidthList->AutoSizeColumns( true );
+ m_gridViaSizeList->SetColMinimalWidth( 1, 150 );
+
+ // Fill cells with actual values:
+ m_gridViaSizeList->SetCellValue( 0, 0, wxEmptyString );
+ m_gridViaSizeList->SetCellValue( 0, 1, wxEmptyString );
+ m_gridTrackWidthList->SetCellValue( 0, 0, wxEmptyString );
+
+ // Give a correct size to row labels column
+ m_gridViaSizeList->SetRowLabelSize( wxGRID_AUTOSIZE );
+ m_gridTrackWidthList->SetRowLabelSize( wxGRID_AUTOSIZE );
+
+ for( unsigned ii = 0; ii < m_TracksWidthList.size(); ii++ )
+ {
+ msg = StringFromValue( g_UserUnit, m_TracksWidthList[ii], false );
+ m_gridTrackWidthList->SetCellValue( ii, 0, msg );
+ }
+
+ for( unsigned ii = 0; ii < m_ViasDimensionsList.size(); ii++ )
+ {
+ msg = StringFromValue( g_UserUnit, m_ViasDimensionsList[ii].m_Diameter, false );
+ m_gridViaSizeList->SetCellValue( ii, 0, msg );
+
+ if( m_ViasDimensionsList[ii].m_Drill > 0 )
+ {
+ msg = StringFromValue( g_UserUnit, m_ViasDimensionsList[ii].m_Drill, false );
+ m_gridViaSizeList->SetCellValue( ii, 1, msg );
+ }
+ }
+}
+
+
+// Sort comparison function (helper for makePointers() )
+static bool sortByClassThenName( NETCUP* a, NETCUP* b )
+{
+ // return a < b
+ if( a->clazz < b->clazz )
+ return true;
+
+ // inside the same class, sort by net name:
+ if( a->clazz == b->clazz )
+ {
+ if( a->net < b->net )
+ return true;
+ }
+
+ return false;
+}
+
+
+void DIALOG_DESIGN_RULES::makePointers( PNETCUPS* aList, const wxString& aNetClassName )
+{
+ aList->clear();
+
+ if( wildCard == aNetClassName )
+ {
+ for( NETCUPS::iterator n = m_AllNets.begin(); n != m_AllNets.end(); ++n )
+ {
+ aList->push_back( &*n );
+ }
+
+ sort( aList->begin(), aList->end(), sortByClassThenName );
+
+ // could use a different sort order for wildCard case.
+ }
+ else
+ {
+ for( NETCUPS::iterator n = m_AllNets.begin(); n != m_AllNets.end(); ++n )
+ {
+ if( n->clazz == aNetClassName )
+ aList->push_back( &*n );
+ }
+
+ sort( aList->begin(), aList->end(), sortByClassThenName );
+ }
+}
+
+
+void DIALOG_DESIGN_RULES::FillListBoxWithNetNames( NETS_LIST_CTRL* aListCtrl,
+ const wxString& aNetClass )
+{
+ aListCtrl->ClearList();
+
+ PNETCUPS ptrList;
+
+ // get a subset of m_AllNets in pointer form, sorted as desired.
+ makePointers( &ptrList, aNetClass );
+
+#if 0 && defined(DEBUG)
+ int r = 0;
+ for( PNETCUPS::iterator i = ptrList.begin(); i!=ptrList.end(); ++i, ++r )
+ {
+ printf( "[%d]: %s %s\n", r, TO_UTF8( (*i)->net ), TO_UTF8( (*i)->clazz ) );
+ }
+
+#endif
+
+ // Add netclass info to m_Netnames and m_Classnames wxArrayString buffers
+ // aListCtrl uses wxLC_VIRTUAL option, so this is fast
+ wxClientDC sDC( aListCtrl );
+ int row = 0;
+ // recompute the column widths here, after setting texts
+ int net_colsize = sDC.GetTextExtent( NET_TITLE ).x;
+ int class_colsize = sDC.GetTextExtent( CLASS_TITLE ).x;
+
+ for( PNETCUPS::iterator i = ptrList.begin(); i!=ptrList.end(); ++i, ++row )
+ {
+ wxSize net_needed = sDC.GetTextExtent( (*i)->net );
+ wxSize class_needed = sDC.GetTextExtent( (*i)->clazz );
+ net_colsize = std::max( net_colsize, net_needed.x );
+ class_colsize = std::max( class_colsize, class_needed.x );
+ aListCtrl->SetRowItems( row, (*i)->net, (*i)->clazz );
+ }
+
+ int margin = sDC.GetTextExtent( wxT( "XX" ) ).x;
+ aListCtrl->SetColumnWidth( 0, net_colsize + margin );
+ aListCtrl->SetColumnWidth( 1, class_colsize + margin );
+ aListCtrl->Refresh();
+}
+
+
+/* Populates combo boxes with the list of existing net classes
+ */
+void DIALOG_DESIGN_RULES::InitializeRulesSelectionBoxes()
+{
+ m_rightClassChoice->Clear();
+ m_leftClassChoice->Clear();
+
+ m_rightClassChoice->Append( wildCard );
+ m_leftClassChoice->Append( wildCard );
+
+ for( int ii = 0; ii < m_grid->GetNumberRows(); ii++ )
+ {
+ m_rightClassChoice->Append( m_grid->GetRowLabelValue( ii ) );
+ m_leftClassChoice->Append( m_grid->GetRowLabelValue( ii ) );
+ }
+
+ m_rightClassChoice->Select( 0 );
+ m_leftClassChoice->Select( 0 );
+
+ m_buttonRightToLeft->Enable( false );
+ m_buttonLeftToRight->Enable( false );
+
+ FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
+ FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
+}
+
+
+/* Initialize the rules list from board
+ */
+
+static void class2gridRow( wxGrid* grid, int row, NETCLASSPTR nc )
+{
+ wxString msg;
+
+ // label is netclass name
+ grid->SetRowLabelValue( row, nc->GetName() );
+
+ msg = StringFromValue( g_UserUnit, nc->GetClearance() );
+ grid->SetCellValue( row, GRID_CLEARANCE, msg );
+
+ msg = StringFromValue( g_UserUnit, nc->GetTrackWidth() );
+ grid->SetCellValue( row, GRID_TRACKSIZE, msg );
+
+ msg = StringFromValue( g_UserUnit, nc->GetViaDiameter() );
+ grid->SetCellValue( row, GRID_VIASIZE, msg );
+
+ msg = StringFromValue( g_UserUnit, nc->GetViaDrill() );
+ grid->SetCellValue( row, GRID_VIADRILL, msg );
+
+ msg = StringFromValue( g_UserUnit, nc->GetuViaDiameter() );
+ grid->SetCellValue( row, GRID_uVIASIZE, msg );
+
+ msg = StringFromValue( g_UserUnit, nc->GetuViaDrill() );
+ grid->SetCellValue( row, GRID_uVIADRILL, msg );
+}
+
+
+void DIALOG_DESIGN_RULES::InitRulesList()
+{
+ NETCLASSES& netclasses = m_BrdSettings->m_NetClasses;
+
+ // the +1 is for the Default NETCLASS.
+ if( netclasses.GetCount() + 1 > (unsigned) m_grid->GetNumberRows() )
+ {
+ m_grid->AppendRows( netclasses.GetCount() + 1 - m_grid->GetNumberRows() );
+ }
+
+ // enter the Default NETCLASS.
+ class2gridRow( m_grid, 0, netclasses.GetDefault() );
+
+ // enter others netclasses
+ int row = 1;
+
+ for( NETCLASSES::iterator i = netclasses.begin(); i!=netclasses.end(); ++i, ++row )
+ {
+ NETCLASSPTR netclass = i->second;
+
+ class2gridRow( m_grid, row, netclass );
+ }
+}
+
+
+static void gridRow2class( wxGrid* grid, int row, NETCLASSPTR nc )
+{
+#define MYCELL( col ) \
+ ValueFromString( g_UserUnit, grid->GetCellValue( row, col ) )
+
+ nc->SetClearance( MYCELL( GRID_CLEARANCE ) );
+ nc->SetTrackWidth( MYCELL( GRID_TRACKSIZE ) );
+ nc->SetViaDiameter( MYCELL( GRID_VIASIZE ) );
+ nc->SetViaDrill( MYCELL( GRID_VIADRILL ) );
+ nc->SetuViaDiameter( MYCELL( GRID_uVIASIZE ) );
+ nc->SetuViaDrill( MYCELL( GRID_uVIADRILL ) );
+}
+
+
+void DIALOG_DESIGN_RULES::CopyRulesListToBoard()
+{
+ NETCLASSES& netclasses = m_BrdSettings->m_NetClasses;
+
+ // Remove all netclasses from board. We'll copy new list after
+ netclasses.Clear();
+
+ // Copy the default NetClass:
+ gridRow2class( m_grid, 0, netclasses.GetDefault() );
+
+ // Copy other NetClasses :
+ for( int row = 1; row < m_grid->GetNumberRows(); ++row )
+ {
+ NETCLASSPTR nc = boost::make_shared<NETCLASS>( m_grid->GetRowLabelValue( row ) );
+
+ if( !m_BrdSettings->m_NetClasses.Add( nc ) )
+ {
+ // this netclass cannot be added because an other netclass with the same name exists
+ // Should not occur because OnAddNetclassClick() tests for existing NetClass names
+ wxString msg;
+ msg.Printf( wxT( "CopyRulesListToBoard(): The NetClass \"%s\" already exists. Skip" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ wxMessageBox( msg );
+
+ continue;
+ }
+
+ gridRow2class( m_grid, row, nc );
+ }
+
+ // Now read all nets and push them in the corresponding netclass net buffer
+ for( NETCUPS::const_iterator netcup = m_AllNets.begin(); netcup != m_AllNets.end(); ++netcup )
+ {
+ NETCLASSPTR nc = netclasses.Find( netcup->clazz );
+ wxASSERT( nc );
+ nc->Add( netcup->net );
+ }
+
+ m_Pcb->SynchronizeNetsAndNetClasses();
+}
+
+
+void DIALOG_DESIGN_RULES::CopyGlobalRulesToBoard()
+{
+ m_BrdSettings->m_BlindBuriedViaAllowed = m_OptViaType->GetSelection() > 0;
+
+ // Update vias minimum values for DRC
+ m_BrdSettings->m_ViasMinSize = ValueFromTextCtrl( *m_SetViasMinSizeCtrl );
+ m_BrdSettings->m_ViasMinDrill = ValueFromTextCtrl( *m_SetViasMinDrillCtrl );
+
+ m_BrdSettings->m_MicroViasAllowed = m_AllowMicroViaCtrl->GetSelection() == 1;
+
+ // Update microvias minimum values for DRC
+ m_BrdSettings->m_MicroViasMinSize = ValueFromTextCtrl( *m_SetMicroViasMinSizeCtrl );
+ m_BrdSettings->m_MicroViasMinDrill = ValueFromTextCtrl( *m_SetMicroViasMinDrillCtrl );
+
+ // Update tracks minimum values for DRC
+ m_BrdSettings->m_TrackMinWidth = ValueFromTextCtrl( *m_SetTrackMinWidthCtrl );
+}
+
+
+void DIALOG_DESIGN_RULES::CopyDimensionsListsToBoard()
+{
+ wxString msg;
+
+ // Reinitialize m_TrackWidthList
+ m_TracksWidthList.clear();
+
+ for( int row = 0; row < m_gridTrackWidthList->GetNumberRows(); ++row )
+ {
+ msg = m_gridTrackWidthList->GetCellValue( row, 0 );
+
+ if( msg.IsEmpty() )
+ continue;
+
+ int value = ValueFromString( g_UserUnit, msg );
+ m_TracksWidthList.push_back( value );
+ }
+
+ // Sort new list by by increasing value
+ sort( m_TracksWidthList.begin(), m_TracksWidthList.end() );
+
+ // Reinitialize m_TrackWidthList
+ m_ViasDimensionsList.clear();
+
+ for( int row = 0; row < m_gridViaSizeList->GetNumberRows(); ++row )
+ {
+ msg = m_gridViaSizeList->GetCellValue( row, 0 );
+
+ if( msg.IsEmpty() )
+ continue;
+
+ int value = ValueFromString( g_UserUnit, msg );
+ VIA_DIMENSION via_dim;
+ via_dim.m_Diameter = value;
+ msg = m_gridViaSizeList->GetCellValue( row, 1 );
+
+ if( !msg.IsEmpty() )
+ {
+ value = ValueFromString( g_UserUnit, msg );
+ via_dim.m_Drill = value;
+ }
+
+ m_ViasDimensionsList.push_back( via_dim );
+ }
+
+ // Sort new list by by increasing value
+ sort( m_ViasDimensionsList.begin(), m_ViasDimensionsList.end() );
+
+ std::vector<int>* tlist = &m_BrdSettings->m_TrackWidthList;
+
+ // Remove old "custom" sizes
+ tlist->erase( tlist->begin() + 1, tlist->end() );
+
+ // Add new "custom" sizes
+ tlist->insert( tlist->end(), m_TracksWidthList.begin(), m_TracksWidthList.end() );
+
+ // Reinitialize m_ViaSizeList
+ std::vector<VIA_DIMENSION>* vialist = &m_BrdSettings->m_ViasDimensionsList;
+ vialist->erase( vialist->begin() + 1, vialist->end() );
+ vialist->insert( vialist->end(), m_ViasDimensionsList.begin(), m_ViasDimensionsList.end() );
+}
+
+
+void DIALOG_DESIGN_RULES::OnNotebookPageChanged( wxNotebookEvent& event )
+{
+ s_LastTabSelection = event.GetSelection();
+
+ // Skip() allows OSX to properly refresh controls.
+ event.Skip();
+}
+
+
+bool DIALOG_DESIGN_RULES::TransferDataFromWindow()
+{
+ if( !wxDialog::TransferDataFromWindow() )
+ return false;
+
+ wxString errorMsg;
+
+ if( !TestDataValidity( &errorMsg ) )
+ {
+ HTML_MESSAGE_BOX dlg( this, _( "Design Rule Setting Error" ) );
+ dlg.MessageSet( errorMsg );
+ dlg.ShowModal();
+ return false;
+ }
+
+ CopyRulesListToBoard();
+ CopyGlobalRulesToBoard();
+ CopyDimensionsListsToBoard();
+ m_BrdSettings->SetCurrentNetClass( NETCLASS::Default );
+ return true;
+}
+
+
+void DIALOG_DESIGN_RULES::OnAddNetclassClick( wxCommandEvent& event )
+{
+ wxString class_name;
+
+ // @todo set validator to ensure net class name is valid rather than all of the checks
+ // after the OK button has been selected.
+ wxTextEntryDialog dlg( this, _( "New Net Class Name:" ), wxEmptyString, class_name );
+
+ if( dlg.ShowModal() != wxID_OK )
+ return; // canceled by user
+
+ class_name = dlg.GetValue();
+ class_name.Trim( true );
+ class_name.Trim( false );
+
+ if( class_name.IsEmpty() )
+ return; // empty name not allowed
+
+ // The name must dot exists:
+ for( int ii = 0; ii < m_grid->GetNumberRows(); ii++ )
+ {
+ wxString value;
+ value = m_grid->GetRowLabelValue( ii );
+
+ if( class_name.CmpNoCase( value ) == 0 ) // Already exists!
+ {
+ DisplayError( this, _( "Duplicate net class names are not allowed." ) );
+ return;
+ }
+ }
+
+ m_grid->AppendRows();
+ m_grid->SetRowLabelValue( m_grid->GetNumberRows() - 1, class_name );
+
+ // Copy values of the default class:
+ int irow = m_grid->GetNumberRows() - 1;
+
+ for( int icol = 0; icol < m_grid->GetNumberCols(); icol++ )
+ {
+ wxString value;
+ value = m_grid->GetCellValue( 0, icol );
+ m_grid->SetCellValue( irow, icol, value );
+ }
+
+ InitializeRulesSelectionBoxes();
+}
+
+
+// Sort function for wxArrayInt. Items (ints) are sorted by decreasing value
+// used in DIALOG_DESIGN_RULES::OnRemoveNetclassClick
+int sort_int( int* first, int* second )
+{
+ return *second - *first;
+}
+
+
+void DIALOG_DESIGN_RULES::OnRemoveNetclassClick( wxCommandEvent& event )
+{
+ wxArrayInt select = m_grid->GetSelectedRows();
+
+ // Sort selection by decreasing index order:
+ select.Sort( sort_int );
+ bool reinit = false;
+
+ // rows labels are not removed when deleting rows: they are not deleted.
+ // So we must store them, remove corresponding labels and reinit them
+ wxArrayString labels;
+
+ for( int ii = 0; ii < m_grid->GetNumberRows(); ii++ )
+ labels.Add( m_grid->GetRowLabelValue( ii ) );
+
+ // Delete rows from last to first (this is the order wxArrayInt select after sorting) )
+ // This order is Ok when removing rows
+ for( unsigned ii = 0; ii < select.GetCount(); ii++ )
+ {
+ int grid_row = select[ii];
+
+ if( grid_row != 0 ) // Do not remove the default class
+ {
+ wxString classname = m_grid->GetRowLabelValue( grid_row );
+ m_grid->DeleteRows( grid_row );
+ labels.RemoveAt( grid_row ); // Remove corresponding row label
+ reinit = true;
+
+ // reset the net class to default for members of the removed class
+ swapNetClass( classname, NETCLASS::Default );
+ }
+ else
+ wxMessageBox( _( "The default net class cannot be removed" ) );
+ }
+
+ if( reinit )
+ {
+ // Reinit labels :
+ for( unsigned ii = 1; ii < labels.GetCount(); ii++ )
+ m_grid->SetRowLabelValue( ii, labels[ii] );
+
+ InitializeRulesSelectionBoxes();
+ }
+}
+
+
+void DIALOG_DESIGN_RULES::OnMoveUpSelectedNetClass( wxCommandEvent& event )
+{
+ // Cannot move up rules if we have 1 or 2 rules only
+ if( m_grid->GetNumberRows() < 3 )
+ return;
+
+ wxArrayInt select = m_grid->GetSelectedRows();
+ bool reinit = false;
+
+ for( unsigned irow = 0; irow < select.GetCount(); irow++ )
+ {
+ int ii = select[irow];
+ if( ii < 2 ) // The default netclass *must* be the first netclass
+ continue; // so we cannot move up line 0 and 1
+
+ // Swap the rule and the previous rule
+ wxString curr_value, previous_value;
+
+ for( int icol = 0; icol < m_grid->GetNumberCols(); icol++ )
+ {
+ reinit = true;
+ curr_value = m_grid->GetCellValue( ii, icol );
+ previous_value = m_grid->GetCellValue( ii - 1, icol );
+ m_grid->SetCellValue( ii, icol, previous_value );
+ m_grid->SetCellValue( ii - 1, icol, curr_value );
+ }
+
+ curr_value = m_grid->GetRowLabelValue( ii );
+ previous_value = m_grid->GetRowLabelValue( ii - 1 );
+ m_grid->SetRowLabelValue( ii, previous_value );
+ m_grid->SetRowLabelValue( ii - 1, curr_value );
+ }
+
+ if( reinit )
+ InitializeRulesSelectionBoxes();
+}
+
+
+void DIALOG_DESIGN_RULES::OnLeftCBSelection( wxCommandEvent& event )
+{
+ FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
+
+ if( m_leftClassChoice->GetStringSelection() == m_rightClassChoice->GetStringSelection() )
+ {
+ m_buttonRightToLeft->Enable( false );
+ m_buttonLeftToRight->Enable( false );
+ }
+ else
+ {
+ m_buttonRightToLeft->Enable( true );
+ m_buttonLeftToRight->Enable( true );
+ }
+}
+
+
+void DIALOG_DESIGN_RULES::OnRightCBSelection( wxCommandEvent& event )
+{
+ FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
+
+ if( m_leftClassChoice->GetStringSelection() == m_rightClassChoice->GetStringSelection() )
+ {
+ m_buttonRightToLeft->Enable( false );
+ m_buttonLeftToRight->Enable( false );
+ }
+ else
+ {
+ m_buttonRightToLeft->Enable( true );
+ m_buttonLeftToRight->Enable( true );
+ }
+}
+
+
+void DIALOG_DESIGN_RULES::moveSelectedItems( NETS_LIST_CTRL* src, const wxString& newClassName )
+{
+ wxListItem item;
+ wxString netName;
+
+ item.m_mask |= wxLIST_MASK_TEXT; // Validate the member m_text of the wxListItem item
+
+ for( int row = 0; row < src->GetItemCount(); ++row )
+ {
+ if( !src->GetItemState( row, wxLIST_STATE_SELECTED ) )
+ continue;
+
+ item.SetColumn( 0 );
+ item.SetId( row );
+
+ src->GetItem( item );
+ netName = item.GetText();
+
+ setNetClass( netName, newClassName == wildCard ? NETCLASS::Default : newClassName );
+ }
+}
+
+
+void DIALOG_DESIGN_RULES::OnRightToLeftCopyButton( wxCommandEvent& event )
+{
+ wxString newClassName = m_leftClassChoice->GetStringSelection();
+
+ moveSelectedItems( m_rightListCtrl, newClassName );
+
+ FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
+ FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
+}
+
+
+void DIALOG_DESIGN_RULES::OnLeftToRightCopyButton( wxCommandEvent& event )
+{
+ wxString newClassName = m_rightClassChoice->GetStringSelection();
+
+ moveSelectedItems( m_leftListCtrl, newClassName );
+
+ FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
+ FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
+}
+
+
+void DIALOG_DESIGN_RULES::OnLeftSelectAllButton( wxCommandEvent& event )
+{
+ for( int ii = 0; ii < m_leftListCtrl->GetItemCount(); ii++ )
+ m_leftListCtrl->SetItemState( ii, wxLIST_STATE_SELECTED, wxLIST_STATE_SELECTED );
+}
+
+
+void DIALOG_DESIGN_RULES::OnRightSelectAllButton( wxCommandEvent& event )
+{
+ for( int ii = 0; ii < m_rightListCtrl->GetItemCount(); ii++ )
+ m_rightListCtrl->SetItemState( ii, wxLIST_STATE_SELECTED, wxLIST_STATE_SELECTED );
+}
+
+
+void DIALOG_DESIGN_RULES::setNetClass( const wxString& aNetName, const wxString& aClassName )
+{
+ for( NETCUPS::iterator i = m_AllNets.begin(); i != m_AllNets.end(); ++i )
+ {
+ if( i->net == aNetName )
+ {
+ i->clazz = aClassName;
+ break;
+ }
+ }
+}
+
+
+bool DIALOG_DESIGN_RULES::TestDataValidity( wxString* aErrorMsg )
+{
+ bool result = true;
+
+ wxString msg;
+ wxString errorMsg;
+
+ int minViaDia = ValueFromTextCtrl( *m_SetViasMinSizeCtrl );
+ int minViaDrill = ValueFromTextCtrl( *m_SetViasMinDrillCtrl );
+ int minUViaDia = ValueFromTextCtrl( *m_SetMicroViasMinSizeCtrl );
+ int minUViaDrill = ValueFromTextCtrl( *m_SetMicroViasMinDrillCtrl );
+ int minTrackWidth = ValueFromTextCtrl( *m_SetTrackMinWidthCtrl );
+ int maxval = 1000 * IU_PER_MILS; // a max value for tracks and vias sizes (1 inch)
+
+ // @todo Change this code to set the focus to the control where the first error occurs
+ // so the user doesn't have to figure out where the issue is.
+
+ // Test net class parameters.
+ for( int row = 0; row < m_grid->GetNumberRows(); row++ )
+ {
+ int tracksize = ValueFromString( g_UserUnit,
+ m_grid->GetCellValue( row, GRID_TRACKSIZE ) );
+ if( tracksize < minTrackWidth )
+ {
+ result = false;
+ msg.Printf( _( "%s: <b>Track Size</b> &lt; <b>Min Track Size</b><br>" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ errorMsg += msg;
+ }
+
+ // Test vias
+ int viadia = ValueFromString( g_UserUnit,
+ m_grid->GetCellValue( row, GRID_VIASIZE ) );
+
+ if( viadia < minViaDia )
+ {
+ result = false;
+ msg.Printf( _( "%s: <b>Via Diameter</b> &lt; <b>Minimum Via Diameter</b><br>" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ errorMsg += msg;
+ }
+
+ int viadrill = ValueFromString( g_UserUnit,
+ m_grid->GetCellValue( row, GRID_VIADRILL ) );
+
+ if( viadrill >= viadia )
+ {
+ result = false;
+ msg.Printf( _( "%s: <b>Via Drill</b> &ge; <b>Via Dia</b><br>" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ errorMsg += msg;
+ }
+
+ if( viadrill < minViaDrill )
+ {
+ result = false;
+ msg.Printf( _( "%s: <b>Via Drill</b> &lt; <b>Min Via Drill</b><br>" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ errorMsg += msg;
+ }
+
+ // Test Micro vias
+ int muviadia = ValueFromString( g_UserUnit,
+ m_grid->GetCellValue( row, GRID_uVIASIZE ) );
+
+ if( muviadia < minUViaDia )
+ {
+ result = false;
+ msg.Printf( _( "%s: <b>MicroVia Diameter</b> &lt; <b>MicroVia Min Diameter</b><br>" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ errorMsg += msg;
+ }
+
+ int muviadrill = ValueFromString( g_UserUnit,
+ m_grid->GetCellValue( row, GRID_uVIADRILL ) );
+
+ if( muviadrill >= muviadia )
+ {
+ result = false;
+ msg.Printf( _( "%s: <b>MicroVia Drill</b> &ge; <b>MicroVia Dia</b><br>" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ errorMsg += msg;
+ }
+
+ if( muviadrill < minUViaDrill )
+ {
+ result = false;
+ msg.Printf( _( "%s: <b>MicroVia Drill</b> &lt; <b>MicroVia Min Drill</b><br>" ),
+ GetChars( m_grid->GetRowLabelValue( row ) ) );
+ errorMsg += msg;
+ }
+ }
+
+ // Test custom tracks
+ for( int row = 0; row < m_gridTrackWidthList->GetNumberRows(); ++row )
+ {
+ wxString tvalue = m_gridTrackWidthList->GetCellValue( row, 0 );
+
+ if( tvalue.IsEmpty() )
+ continue;
+
+ int tracksize = ValueFromString( g_UserUnit, tvalue );
+
+ if( tracksize < minTrackWidth )
+ {
+ result = false;
+ msg.Printf( _( "<b>Extra Track %d Size</b> %s &lt; <b>Min Track Size</b><br>" ),
+ row + 1, GetChars( tvalue ) );
+
+ errorMsg += msg;
+ }
+
+ if( tracksize > maxval )
+ {
+ result = false;
+ msg.Printf( _( "<b>Extra Track %d Size</b> %s &gt; <b>1 inch!</b><br>" ),
+ row + 1, GetChars( tvalue ) );
+
+ errorMsg += msg;
+ }
+ }
+
+ // Test custom vias
+ for( int row = 0; row < m_gridViaSizeList->GetNumberRows(); ++row )
+ {
+ wxString tvalue = m_gridViaSizeList->GetCellValue( row, 0 );
+
+ if( tvalue.IsEmpty() )
+ continue;
+
+ int viadia = ValueFromString( g_UserUnit, tvalue );
+
+ if( viadia < minViaDia )
+ {
+ result = false;
+ msg.Printf( _( "<b>Extra Via %d Size</b> %s &lt; <b>Min Via Size</b><br>" ),
+ row + 1, GetChars( tvalue ) );
+ errorMsg += msg;
+ }
+
+ wxString drlvalue = m_gridViaSizeList->GetCellValue( row, 1 );
+
+ if( drlvalue.IsEmpty() )
+ {
+ result = false;
+ msg.Printf( _( "<b>No via drill size define in row %d</b><br>" ), row + 1 );
+ errorMsg += msg;
+ continue;
+ }
+
+ int viadrill = ValueFromString( g_UserUnit, drlvalue );
+
+ if( viadrill < minViaDrill )
+ {
+ result = false;
+ msg.Printf( _( "<b>Extra Via %d Drill</b> %s &lt; <b>Min Via Drill %s</b><br>" ),
+ row + 1, GetChars( drlvalue ),
+ GetChars( m_SetViasMinDrillCtrl->GetValue() ) );
+ errorMsg += msg;
+ }
+
+ if( viadia <= viadrill )
+ {
+ result = false;
+ msg.Printf( _( "<b>Extra Via %d Size</b> %s &le; <b> Drill Size</b> %s<br>" ),
+ row + 1, GetChars( tvalue ), GetChars( drlvalue ) );
+ errorMsg += msg;
+ }
+
+ // Test for a reasonable via size:
+ if( viadia > maxval ) // 1 inch!
+ {
+ result = false;
+ msg.Printf( _( "<b>Extra Via %d Size</b>%s &gt; <b>1 inch!</b><br>" ),
+ row + 1, GetChars( tvalue ) );
+ errorMsg += msg;
+ }
+ }
+
+ if( !result && aErrorMsg )
+ *aErrorMsg = errorMsg;
+
+ return result;
+}