2009-07-18 11:44:19 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Name: dialog_design_rules.cpp
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2009-09-11 06:18:55 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KICAD, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2004-2009 Jean-Pierre Charras, jean-pierre.charras@inpg.fr
|
|
|
|
* Copyright (C) 2009 Dick Hollenbeck, dick@softplc.com
|
|
|
|
* Copyright (C) 2009 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
/* functions relatives to the design rules editor
|
|
|
|
*/
|
|
|
|
#include "fctsys.h"
|
|
|
|
#include "common.h"
|
|
|
|
#include "class_drawpanel.h"
|
|
|
|
|
|
|
|
#include "confirm.h"
|
|
|
|
#include "pcbnew.h"
|
2009-07-30 11:04:07 +00:00
|
|
|
#include "wxPcbStruct.h"
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-22 12:27:57 +00:00
|
|
|
#include "pcbnew_id.h"
|
2009-07-18 11:44:19 +00:00
|
|
|
#include "dialog_design_rules.h"
|
|
|
|
#include "wx/generic/gridctrl.h"
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
// 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)");
|
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
|
|
|
|
/***********************************************************************************/
|
|
|
|
DIALOG_DESIGN_RULES::DIALOG_DESIGN_RULES( WinEDA_PcbFrame* parent ) :
|
|
|
|
DIALOG_DESIGN_RULES_BASE( parent )
|
|
|
|
/***********************************************************************************/
|
|
|
|
{
|
|
|
|
m_Parent = parent;
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
wxListItem column0;
|
|
|
|
wxListItem column1;
|
|
|
|
|
|
|
|
column0.Clear();
|
|
|
|
column1.Clear();
|
|
|
|
|
|
|
|
column0.SetImage( -1 );
|
|
|
|
column1.SetImage( -1 );
|
|
|
|
|
|
|
|
column0.SetText( _( "Net" ) );
|
|
|
|
column1.SetText( _( "Class" ) );
|
|
|
|
|
|
|
|
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 );
|
|
|
|
|
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
Init();
|
|
|
|
SetAutoLayout( true );
|
|
|
|
GetSizer()->Fit( this );
|
|
|
|
GetSizer()->SetSizeHints( this );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-23 05:53:12 +00:00
|
|
|
/* Display on m_MessagesList the current global settings:
|
|
|
|
* minimal values for tracks, vias, clearance ...
|
|
|
|
*/
|
|
|
|
void DIALOG_DESIGN_RULES::PrintCurrentSettings( )
|
|
|
|
{
|
|
|
|
wxString msg, value;
|
|
|
|
int internal_units = m_Parent->m_InternalUnits;
|
|
|
|
|
2009-09-29 04:53:02 +00:00
|
|
|
m_MessagesList->AppendToPage(_("<b>Current general settings:</b><br>") );
|
2009-09-23 05:53:12 +00:00
|
|
|
|
|
|
|
// Display min values:
|
|
|
|
value = ReturnStringFromValue( g_UnitMetric, g_DesignSettings.m_TrackMinWidth, internal_units, true );
|
2009-09-29 04:53:02 +00:00
|
|
|
msg.Printf(_("Minimum value for tracks width: <b>%s</b><br>\n"), GetChars( value ) );
|
2009-09-23 05:53:12 +00:00
|
|
|
m_MessagesList->AppendToPage(msg);
|
2009-09-29 04:53:02 +00:00
|
|
|
|
2009-09-23 05:53:12 +00:00
|
|
|
value = ReturnStringFromValue( g_UnitMetric, g_DesignSettings.m_ViasMinSize, internal_units, true );
|
2009-09-29 04:53:02 +00:00
|
|
|
msg.Printf(_("Minimum value for vias diameter: <b>%s</b><br>\n"), GetChars( value ) );
|
2009-09-23 05:53:12 +00:00
|
|
|
m_MessagesList->AppendToPage(msg);
|
2009-09-29 04:53:02 +00:00
|
|
|
|
2009-09-23 05:53:12 +00:00
|
|
|
value = ReturnStringFromValue( g_UnitMetric, g_DesignSettings.m_MicroViasMinSize, internal_units, true );
|
2009-09-29 04:53:02 +00:00
|
|
|
msg.Printf(_("Minimum value for microvias diameter: <b>%s</b><br>\n"), GetChars( value ) );
|
2009-09-23 05:53:12 +00:00
|
|
|
m_MessagesList->AppendToPage(msg);
|
2009-09-29 04:53:02 +00:00
|
|
|
|
2009-09-23 05:53:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
/********************************************************************/
|
|
|
|
void DIALOG_DESIGN_RULES::Init()
|
|
|
|
/********************************************************************/
|
|
|
|
{
|
|
|
|
SetFocus();
|
2009-07-18 16:06:48 +00:00
|
|
|
SetReturnCode( 0 );
|
2009-07-18 11:44:19 +00:00
|
|
|
|
|
|
|
// Initialize the layers grid:
|
2009-07-18 16:06:48 +00:00
|
|
|
m_Pcb = m_Parent->GetBoard();
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// Initialize the Rules List
|
|
|
|
InitRulesList();
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// copy all NETs into m_AllNets by adding them as NETCUPs.
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 17:28:38 +00:00
|
|
|
// @todo go fix m_Pcb->SynchronizeNetsAndNetClasses() so that the netcode==0 is not present in the BOARD::m_NetClasses
|
|
|
|
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
NETCLASS* netclass;
|
|
|
|
|
|
|
|
NETCLASSES& netclasses = m_Pcb->m_NetClasses;
|
|
|
|
|
|
|
|
netclass = netclasses.GetDefault();
|
|
|
|
|
2009-09-23 05:53:12 +00:00
|
|
|
// Initialize list of nets for Default Net Class
|
2009-09-10 15:22:26 +00:00
|
|
|
for( NETCLASS::const_iterator name = netclass->begin(); name != netclass->end(); ++name )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
m_AllNets.push_back( NETCUP( *name, netclass->GetName() ) );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-09-23 05:53:12 +00:00
|
|
|
// Initialize list of nets for others (custom) Net Classes
|
2009-09-10 15:22:26 +00:00
|
|
|
for( NETCLASSES::const_iterator nc = netclasses.begin(); nc != netclasses.end(); ++nc )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
netclass = nc->second;
|
|
|
|
|
|
|
|
for( NETCLASS::const_iterator name = netclass->begin(); name != netclass->end(); ++name )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
m_AllNets.push_back( NETCUP( *name, netclass->GetName() ) );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
InitializeRulesSelectionBoxes();
|
2009-09-29 04:53:02 +00:00
|
|
|
|
2009-09-23 05:53:12 +00:00
|
|
|
PrintCurrentSettings( );
|
2009-09-10 15:22:26 +00:00
|
|
|
}
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// Sort comparison function
|
|
|
|
static bool sortByClassThenName( NETCUP* a, NETCUP* b )
|
|
|
|
{
|
|
|
|
// return a < b
|
|
|
|
|
|
|
|
if( a->clazz < b->clazz )
|
|
|
|
return true;
|
|
|
|
|
2009-09-10 17:28:38 +00:00
|
|
|
if( a->net < b->net )
|
2009-09-10 15:22:26 +00:00
|
|
|
return true;
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DIALOG_DESIGN_RULES::makePointers( PNETCUPS* aList, const wxString& aNetClassName )
|
|
|
|
{
|
|
|
|
aList->clear();
|
|
|
|
|
|
|
|
if( wildCard == aNetClassName )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
for( NETCUPS::iterator n = m_AllNets.begin(); n != m_AllNets.end(); ++n )
|
|
|
|
{
|
|
|
|
aList->push_back( &*n );
|
|
|
|
}
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
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 )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
if( n->clazz == aNetClassName )
|
|
|
|
aList->push_back( &*n );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
sort( aList->begin(), aList->end(), sortByClassThenName );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
2009-09-10 15:22:26 +00:00
|
|
|
}
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
void DIALOG_DESIGN_RULES::setRowItem( wxListCtrl* aListCtrl, int aRow, NETCUP* aNetAndClass )
|
|
|
|
{
|
|
|
|
wxASSERT( aRow >= 0 );
|
|
|
|
|
|
|
|
// insert blanks if aRow is larger than existing count
|
|
|
|
while( aRow >= aListCtrl->GetItemCount() )
|
|
|
|
{
|
|
|
|
long ndx = aListCtrl->InsertItem( aListCtrl->GetItemCount(), wxEmptyString );
|
|
|
|
|
|
|
|
wxASSERT( ndx >= 0 );
|
|
|
|
|
|
|
|
aListCtrl->SetItem( ndx, 1, wxEmptyString );
|
|
|
|
}
|
|
|
|
|
|
|
|
aListCtrl->SetItem( aRow, 0, aNetAndClass->net );
|
|
|
|
aListCtrl->SetItem( aRow, 1, aNetAndClass->clazz );
|
|
|
|
|
|
|
|
// recompute the column widths here, after setting texts
|
|
|
|
aListCtrl->SetColumnWidth( 0, wxLIST_AUTOSIZE );
|
|
|
|
aListCtrl->SetColumnWidth( 1, wxLIST_AUTOSIZE );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
/**
|
|
|
|
* Function FillListBoxWithNetNames
|
2009-09-10 15:30:30 +00:00
|
|
|
* populates aListCtrl with net names and class names from m_AllNets in a two column display.
|
2009-07-18 11:44:19 +00:00
|
|
|
*/
|
2009-09-10 15:22:26 +00:00
|
|
|
void DIALOG_DESIGN_RULES::FillListBoxWithNetNames( wxListCtrl* aListCtrl, const wxString& aNetClass )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 17:28:38 +00:00
|
|
|
aListCtrl->DeleteAllItems();
|
2009-09-10 15:30:30 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
PNETCUPS ptrList;
|
|
|
|
|
2009-09-10 15:30:30 +00:00
|
|
|
// get a subset of m_AllNets in pointer form, sorted as desired.
|
2009-09-10 15:22:26 +00:00
|
|
|
makePointers( &ptrList, aNetClass );
|
|
|
|
|
2009-09-10 17:28:38 +00:00
|
|
|
#if 0 && defined(DEBUG)
|
2009-09-10 15:22:26 +00:00
|
|
|
int r = 0;
|
|
|
|
for( PNETCUPS::iterator i = ptrList.begin(); i!=ptrList.end(); ++i, ++r )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
printf("[%d]: %s %s\n", r, CONV_TO_UTF8( (*i)->net ), CONV_TO_UTF8( (*i)->clazz ) );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
2009-09-10 15:22:26 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// to speed up inserting we hide the control temporarily
|
|
|
|
aListCtrl->Hide();
|
|
|
|
|
|
|
|
int row = 0;
|
|
|
|
for( PNETCUPS::iterator i = ptrList.begin(); i!=ptrList.end(); ++i, ++row )
|
|
|
|
{
|
|
|
|
setRowItem( aListCtrl, row, *i );
|
|
|
|
}
|
|
|
|
|
|
|
|
aListCtrl->Show();
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-11 06:18:55 +00:00
|
|
|
/* Initialize the combo boxes by the list of existing net classes
|
2009-07-18 11:44:19 +00:00
|
|
|
*/
|
|
|
|
void DIALOG_DESIGN_RULES::InitializeRulesSelectionBoxes()
|
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
m_rightClassChoice->Clear();
|
|
|
|
m_leftClassChoice->Clear();
|
|
|
|
|
|
|
|
m_rightClassChoice->Append( wildCard );
|
|
|
|
m_leftClassChoice->Append( wildCard );
|
|
|
|
|
|
|
|
for( int ii = 0; ii < m_grid->GetNumberRows(); ii++ )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
m_rightClassChoice->Append( m_grid->GetRowLabelValue( ii ) );
|
|
|
|
m_leftClassChoice->Append( m_grid->GetRowLabelValue( ii ) );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
m_rightClassChoice->Select( 0 );
|
|
|
|
m_leftClassChoice->Select( 0 );
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
m_buttonRightToLeft->Enable( false );
|
|
|
|
m_buttonLeftToRight->Enable( false );;
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
|
|
|
|
FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
/* Initialize the rules list from board
|
2009-07-18 11:44:19 +00:00
|
|
|
*/
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
static void class2gridRow( wxGrid* grid, int row, NETCLASS* nc, int units )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
wxString msg;
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// label is netclass name
|
|
|
|
grid->SetRowLabelValue( row, nc->GetName() );
|
|
|
|
|
|
|
|
msg = ReturnStringFromValue( g_UnitMetric, nc->GetClearance(), units );
|
|
|
|
grid->SetCellValue( row, GRID_CLEARANCE, msg );
|
|
|
|
|
|
|
|
msg = ReturnStringFromValue( g_UnitMetric, nc->GetTrackWidth(), units );
|
|
|
|
grid->SetCellValue( row, GRID_TRACKSIZE, msg );
|
|
|
|
|
|
|
|
msg = ReturnStringFromValue( g_UnitMetric, nc->GetViaDiameter(), units );
|
|
|
|
grid->SetCellValue( row, GRID_VIASIZE, msg );
|
|
|
|
|
|
|
|
msg = ReturnStringFromValue( g_UnitMetric, nc->GetViaDrill(), units );
|
|
|
|
grid->SetCellValue( row, GRID_VIADRILL, msg );
|
|
|
|
|
|
|
|
msg = ReturnStringFromValue( g_UnitMetric, nc->GetuViaDiameter(), units );
|
|
|
|
grid->SetCellValue( row, GRID_uVIASIZE, msg );
|
|
|
|
|
|
|
|
msg = ReturnStringFromValue( g_UnitMetric, nc->GetuViaDrill(), units );
|
|
|
|
grid->SetCellValue( row, GRID_uVIADRILL, msg );
|
|
|
|
}
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-17 17:48:40 +00:00
|
|
|
/** Function InitRulesList()
|
|
|
|
* Fill the grid showing current rules with values
|
|
|
|
*/
|
2009-07-18 11:44:19 +00:00
|
|
|
void DIALOG_DESIGN_RULES::InitRulesList()
|
|
|
|
{
|
2009-08-17 02:59:38 +00:00
|
|
|
NETCLASSES& netclasses = m_Pcb->m_NetClasses;
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// the +1 is for the Default NETCLASS.
|
|
|
|
if( netclasses.GetCount()+1 > (unsigned) m_grid->GetNumberRows() )
|
|
|
|
{
|
|
|
|
m_grid->AppendRows( netclasses.GetCount()+1 - m_grid->GetNumberRows() );
|
|
|
|
}
|
|
|
|
|
2009-09-17 17:48:40 +00:00
|
|
|
// enter the Default NETCLASS.
|
2009-09-10 15:22:26 +00:00
|
|
|
class2gridRow( m_grid, 0, netclasses.GetDefault(), m_Parent->m_InternalUnits );
|
|
|
|
|
2009-09-17 17:48:40 +00:00
|
|
|
// enter others netclasses
|
2009-09-10 15:22:26 +00:00
|
|
|
int row = 1;
|
|
|
|
for( NETCLASSES::iterator i=netclasses.begin(); i!=netclasses.end(); ++i, ++row )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-08-17 02:59:38 +00:00
|
|
|
NETCLASS* netclass = i->second;
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
class2gridRow( m_grid, row, netclass, m_Parent->m_InternalUnits );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
static void gridRow2class( wxGrid* grid, int row, NETCLASS* nc, int units )
|
|
|
|
{
|
|
|
|
#define MYCELL(col) \
|
|
|
|
ReturnValueFromString( g_UnitMetric, grid->GetCellValue( row, col ), units )
|
|
|
|
|
|
|
|
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 ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-17 17:48:40 +00:00
|
|
|
/* Copy the rules list from grid to board
|
2009-07-18 16:06:48 +00:00
|
|
|
*/
|
2009-07-18 11:44:19 +00:00
|
|
|
void DIALOG_DESIGN_RULES::CopyRulesListToBoard()
|
|
|
|
{
|
2009-08-17 02:59:38 +00:00
|
|
|
NETCLASSES& netclasses = m_Pcb->m_NetClasses;
|
|
|
|
|
2009-09-28 16:14:45 +00:00
|
|
|
// Remove all netclasses from board. We'll copy new list after
|
2009-08-17 02:59:38 +00:00
|
|
|
netclasses.Clear();
|
|
|
|
|
2009-09-28 16:14:45 +00:00
|
|
|
// Copy the default NetClass:
|
2009-09-10 15:22:26 +00:00
|
|
|
gridRow2class( m_grid, 0, netclasses.GetDefault(), m_Parent->m_InternalUnits );
|
|
|
|
|
2009-09-28 16:14:45 +00:00
|
|
|
// Copy other NetClasses :
|
2009-09-10 15:22:26 +00:00
|
|
|
for( int row = 1; row < m_grid->GetNumberRows(); ++row )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
NETCLASS* nc = new NETCLASS( m_Pcb, m_grid->GetRowLabelValue( row ) );
|
|
|
|
|
|
|
|
if( !m_Pcb->m_NetClasses.Add( nc ) )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-28 16:14:45 +00:00
|
|
|
// 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"),
|
2009-09-29 04:53:02 +00:00
|
|
|
GetChars( m_grid->GetRowLabelValue( row ) ) );
|
2009-09-28 16:14:45 +00:00
|
|
|
wxMessageBox( msg );
|
2009-09-10 15:22:26 +00:00
|
|
|
delete nc;
|
|
|
|
continue;
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
2009-08-17 02:59:38 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
gridRow2class( m_grid, row, nc, m_Parent->m_InternalUnits );
|
|
|
|
}
|
|
|
|
|
2009-09-28 16:14:45 +00:00
|
|
|
// 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 )
|
2009-09-10 15:22:26 +00:00
|
|
|
{
|
|
|
|
NETCLASS* nc = netclasses.Find( netcup->clazz );
|
|
|
|
wxASSERT( nc );
|
|
|
|
nc->Add( netcup->net );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-08-17 02:59:38 +00:00
|
|
|
m_Pcb->SynchronizeNetsAndNetClasses();
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
/*****************************************************************/
|
|
|
|
void DIALOG_DESIGN_RULES::OnCancelButtonClick( wxCommandEvent& event )
|
|
|
|
/*****************************************************************/
|
|
|
|
{
|
|
|
|
EndModal( 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
void DIALOG_DESIGN_RULES::OnOkButtonClick( wxCommandEvent& event )
|
|
|
|
/**************************************************************************/
|
|
|
|
{
|
2009-07-18 16:06:48 +00:00
|
|
|
if( !TestDataValidity() )
|
|
|
|
{
|
|
|
|
DisplayError( this, _( "Errors detected, Abort" ) );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
CopyRulesListToBoard();
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
EndModal( wxID_OK );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
void DIALOG_DESIGN_RULES::OnAddNetclassClick( wxCommandEvent& event )
|
|
|
|
/**************************************************************************/
|
|
|
|
{
|
|
|
|
wxString class_name;
|
2009-07-18 16:06:48 +00:00
|
|
|
|
|
|
|
if( Get_Message( _( "New Net Class Name:" ),
|
2009-07-18 11:44:19 +00:00
|
|
|
wxEmptyString,
|
|
|
|
class_name,
|
|
|
|
this ) )
|
|
|
|
return;
|
|
|
|
|
|
|
|
// The name must dot exists:
|
2009-09-10 15:22:26 +00:00
|
|
|
for( int ii = 0; ii < m_grid->GetNumberRows(); ii++ )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
|
|
|
wxString value;
|
2009-09-10 15:22:26 +00:00
|
|
|
value = m_grid->GetRowLabelValue( ii );
|
2009-07-18 16:06:48 +00:00
|
|
|
if( class_name.CmpNoCase( value ) == 0 ) // Already exists!
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-07-18 16:06:48 +00:00
|
|
|
DisplayError( this, _( "This NetClass is already existing, cannot add it; Aborted" ) );
|
2009-07-18 11:44:19 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
m_grid->AppendRows();
|
|
|
|
m_grid->SetRowLabelValue(
|
|
|
|
m_grid->GetNumberRows() - 1,
|
2009-07-18 16:06:48 +00:00
|
|
|
class_name );
|
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
// Copy values of the previous class:
|
2009-09-10 15:22:26 +00:00
|
|
|
int irow = m_grid->GetNumberRows() - 1;
|
|
|
|
for( int icol = 0; icol < m_grid->GetNumberCols(); icol++ )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
|
|
|
wxString value;
|
2009-09-10 15:22:26 +00:00
|
|
|
value = m_grid->GetCellValue( irow - 1, icol );
|
|
|
|
m_grid->SetCellValue( irow, icol, value );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
InitializeRulesSelectionBoxes();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
void DIALOG_DESIGN_RULES::OnRemoveNetclassClick( wxCommandEvent& event )
|
|
|
|
/**************************************************************************/
|
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
wxArrayInt select = m_grid->GetSelectedRows();
|
2009-07-18 16:06:48 +00:00
|
|
|
|
|
|
|
for( int ii = select.GetCount() - 1; ii >= 0; ii-- )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-28 16:14:45 +00:00
|
|
|
int grid_row = select[ii];
|
|
|
|
if( grid_row != 0 ) // Do not remove the default class
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-28 16:14:45 +00:00
|
|
|
wxString classname = m_grid->GetRowLabelValue( grid_row );
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-28 16:14:45 +00:00
|
|
|
m_grid->DeleteRows( grid_row );
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// reset the net class to default for members of the removed class
|
|
|
|
swapNetClass( classname, NETCLASS::Default );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
}
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
InitializeRulesSelectionBoxes();
|
|
|
|
}
|
|
|
|
|
2009-09-17 17:48:40 +00:00
|
|
|
/*
|
|
|
|
* Called on "Move Up" button click
|
|
|
|
* the selected(s) rules are moved up
|
|
|
|
* The default netclass is always the first rule
|
|
|
|
*/
|
|
|
|
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 );
|
|
|
|
}
|
2009-09-29 04:53:02 +00:00
|
|
|
|
2009-09-17 17:48:40 +00:00
|
|
|
if( reinit )
|
|
|
|
InitializeRulesSelectionBoxes();
|
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
/*
|
|
|
|
* Called on the left Choice Box selection
|
2009-07-18 16:06:48 +00:00
|
|
|
*/
|
2009-07-18 11:44:19 +00:00
|
|
|
void DIALOG_DESIGN_RULES::OnLeftCBSelection( wxCommandEvent& event )
|
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
|
|
|
|
if( m_leftClassChoice->GetStringSelection() == m_rightClassChoice->GetStringSelection() )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-07-18 16:06:48 +00:00
|
|
|
m_buttonRightToLeft->Enable( false );
|
|
|
|
m_buttonLeftToRight->Enable( false );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-07-18 16:06:48 +00:00
|
|
|
m_buttonRightToLeft->Enable( true );
|
|
|
|
m_buttonLeftToRight->Enable( true );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
/*
|
|
|
|
* Called on the Right Choice Box selection
|
2009-07-18 16:06:48 +00:00
|
|
|
*/
|
2009-07-18 11:44:19 +00:00
|
|
|
void DIALOG_DESIGN_RULES::OnRightCBSelection( wxCommandEvent& event )
|
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
|
|
|
|
if( m_leftClassChoice->GetStringSelection() == m_rightClassChoice->GetStringSelection() )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-07-18 16:06:48 +00:00
|
|
|
m_buttonRightToLeft->Enable( false );
|
|
|
|
m_buttonLeftToRight->Enable( false );;
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-07-18 16:06:48 +00:00
|
|
|
m_buttonRightToLeft->Enable( true );
|
|
|
|
m_buttonLeftToRight->Enable( true );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-10 17:28:38 +00:00
|
|
|
void DIALOG_DESIGN_RULES::moveSelectedItems( wxListCtrl* src, const wxString& newClassName )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 17:28:38 +00:00
|
|
|
wxListItem item;
|
|
|
|
wxString netName;
|
2009-09-11 06:18:55 +00:00
|
|
|
item.m_mask |= wxLIST_MASK_TEXT ; // Validate the member m_text of the wxListItem item
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2009-09-10 17:28:38 +00:00
|
|
|
for( int row = 0; row < src->GetItemCount(); ++row )
|
2009-07-18 11:44:19 +00:00
|
|
|
{
|
2009-09-10 17:28:38 +00:00
|
|
|
if( !src->GetItemState( row, wxLIST_STATE_SELECTED ) )
|
2009-07-18 11:44:19 +00:00
|
|
|
continue;
|
|
|
|
|
2009-09-10 17:28:38 +00:00
|
|
|
item.SetColumn( 0 );
|
|
|
|
item.SetId( row );
|
|
|
|
|
|
|
|
src->GetItem( item );
|
|
|
|
netName = item.GetText();
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
setNetClass( netName, newClassName == wildCard ? NETCLASS::Default : newClassName );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
2009-09-10 17:28:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DIALOG_DESIGN_RULES::OnRightToLeftCopyButton( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
wxString newClassName = m_leftClassChoice->GetStringSelection();
|
|
|
|
|
|
|
|
moveSelectedItems( m_rightListCtrl, newClassName );
|
2009-07-18 11:44:19 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
|
|
|
|
FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DIALOG_DESIGN_RULES::OnLeftToRightCopyButton( wxCommandEvent& event )
|
|
|
|
{
|
2009-09-10 17:28:38 +00:00
|
|
|
wxString newClassName = m_rightClassChoice->GetStringSelection();
|
|
|
|
|
|
|
|
moveSelectedItems( m_leftListCtrl, newClassName );
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
FillListBoxWithNetNames( m_leftListCtrl, m_leftClassChoice->GetStringSelection() );
|
|
|
|
FillListBoxWithNetNames( m_rightListCtrl, m_rightClassChoice->GetStringSelection() );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
/* Called on clicking the left "select all" button:
|
|
|
|
* select alls items of the left netname list lisxt box
|
2009-07-18 16:06:48 +00:00
|
|
|
*/
|
2009-07-18 11:44:19 +00:00
|
|
|
void DIALOG_DESIGN_RULES::OnLeftSelectAllButton( wxCommandEvent& event )
|
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
for( int ii = 0; ii < m_leftListCtrl->GetItemCount(); ii++ )
|
|
|
|
m_leftListCtrl->SetItemState( ii, wxLIST_STATE_SELECTED, wxLIST_STATE_SELECTED );
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-07-18 11:44:19 +00:00
|
|
|
/* Called on clicking the right "select all" button:
|
|
|
|
* select alls items of the right netname list lisxt box
|
2009-07-18 16:06:48 +00:00
|
|
|
*/
|
2009-07-18 11:44:19 +00:00
|
|
|
void DIALOG_DESIGN_RULES::OnRightSelectAllButton( wxCommandEvent& event )
|
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
2009-07-18 11:44:19 +00:00
|
|
|
}
|
|
|
|
|
2009-07-18 16:06:48 +00:00
|
|
|
|
|
|
|
/* TestDataValidity
|
|
|
|
* Performs a control of data validity
|
|
|
|
* set the background of a bad cell in RED and display an info message
|
|
|
|
* @return true if Ok, false if error
|
|
|
|
*/
|
|
|
|
bool DIALOG_DESIGN_RULES::TestDataValidity()
|
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
bool result = true;
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
m_MessagesList->SetPage(wxEmptyString); // Clear message list
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
wxString msg;
|
2009-07-18 16:06:48 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
for( int row = 0; row < m_grid->GetNumberRows(); row++ )
|
2009-07-18 16:06:48 +00:00
|
|
|
{
|
2009-09-28 16:14:45 +00:00
|
|
|
int tracksize = ReturnValueFromString( g_UnitMetric,
|
|
|
|
m_grid->GetCellValue( row, GRID_TRACKSIZE ),
|
|
|
|
m_Parent->m_InternalUnits );
|
|
|
|
if( tracksize < g_DesignSettings.m_TrackMinWidth )
|
|
|
|
{
|
|
|
|
result = false;
|
|
|
|
msg.Printf( _( "%s: <b>Track Size</b> < <b>Min Track Size</b><br>" ),
|
|
|
|
GetChars( m_grid->GetRowLabelValue(row)) );
|
|
|
|
|
|
|
|
m_MessagesList->AppendToPage( msg );
|
|
|
|
}
|
2009-09-29 04:53:02 +00:00
|
|
|
|
2009-09-28 16:14:45 +00:00
|
|
|
// Test vias
|
2009-09-10 15:22:26 +00:00
|
|
|
int viadia = ReturnValueFromString( g_UnitMetric,
|
2009-09-28 16:14:45 +00:00
|
|
|
m_grid->GetCellValue( row, GRID_VIASIZE ),
|
2009-07-18 16:06:48 +00:00
|
|
|
m_Parent->m_InternalUnits );
|
|
|
|
|
2009-09-28 16:14:45 +00:00
|
|
|
if( viadia < g_DesignSettings.m_ViasMinSize )
|
|
|
|
{
|
|
|
|
result = false;
|
|
|
|
msg.Printf( _( "%s: <b>Via Diameter</b> < <b>Minimun Via Diameter</b><br>" ),
|
|
|
|
GetChars( m_grid->GetRowLabelValue(row)) );
|
|
|
|
|
|
|
|
m_MessagesList->AppendToPage( msg );
|
|
|
|
}
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
int viadrill = ReturnValueFromString( g_UnitMetric,
|
2009-09-28 16:14:45 +00:00
|
|
|
m_grid->GetCellValue( row, GRID_VIADRILL ),
|
2009-07-18 16:06:48 +00:00
|
|
|
m_Parent->m_InternalUnits );
|
2009-09-10 15:22:26 +00:00
|
|
|
if( viadrill && viadrill >= viadia )
|
2009-07-18 16:06:48 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
result = false;
|
|
|
|
msg.Printf( _( "%s: <b>Via Drill</b> ≥ <b>Via Dia</b><br>" ),
|
|
|
|
GetChars( m_grid->GetRowLabelValue(row)) );
|
|
|
|
|
|
|
|
m_MessagesList->AppendToPage( msg );
|
2009-07-18 16:06:48 +00:00
|
|
|
}
|
2009-09-28 16:14:45 +00:00
|
|
|
|
|
|
|
// Test Micro vias
|
|
|
|
int muviadia = ReturnValueFromString( g_UnitMetric,
|
|
|
|
m_grid->GetCellValue( row, GRID_uVIASIZE ),
|
|
|
|
m_Parent->m_InternalUnits );
|
|
|
|
|
|
|
|
if( muviadia < g_DesignSettings.m_MicroViasMinSize )
|
|
|
|
{
|
|
|
|
result = false;
|
|
|
|
msg.Printf( _( "%s: <b>MicroVia Diameter</b> < <b>Minimun MicroVia Diameter</b><br>" ),
|
|
|
|
GetChars( m_grid->GetRowLabelValue(row)) );
|
|
|
|
|
|
|
|
m_MessagesList->AppendToPage( msg );
|
|
|
|
}
|
|
|
|
|
|
|
|
int muviadrill = ReturnValueFromString( g_UnitMetric,
|
|
|
|
m_grid->GetCellValue( row, GRID_uVIADRILL ),
|
|
|
|
m_Parent->m_InternalUnits );
|
|
|
|
if( muviadrill && muviadrill >= muviadia )
|
|
|
|
{
|
|
|
|
result = false;
|
|
|
|
msg.Printf( _( "%s: <b>MicroVia Drill</b> ≥ <b>MicroVia Dia</b><br>" ),
|
|
|
|
GetChars( m_grid->GetRowLabelValue(row)) );
|
|
|
|
|
|
|
|
m_MessagesList->AppendToPage( msg );
|
|
|
|
}
|
2009-07-18 16:06:48 +00:00
|
|
|
}
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
return result;
|
2009-07-18 16:06:48 +00:00
|
|
|
}
|