kicad/pcbnew/dialogs/dialog_board_reannotate.cpp

966 lines
33 KiB
C++
Raw Normal View History

2020-04-21 01:44:17 +00:00
/*
* This program source code file is part of KiCad, a free EDA CAD application.
*
* Copyright (C) 2020 Brian Piccioni brian@documenteddesigns.com
* Copyright (C) 1992-2020 KiCad Developers, see AUTHORS.txt for contributors.
* @author Brian Piccioni <brian@documenteddesigns.com>
*
* This program is free software; you can redistribute it and/or
2020-04-21 01:44:17 +00:00
* 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
*/
#include <base_units.h>
#include <bitmaps.h>
#include <board_commit.h>
#include <confirm.h>
#include <ctype.h>
#include <dialog_board_reannotate.h>
#include <fstream>
#include <kiface_i.h>
#include <mail_type.h>
#include <pcbnew_settings.h>
#include <sstream>
#include <tool/tool_manager.h>
#include <tool/grid_menu.h>
bool SortYFirst;
bool DescendingFirst;
bool DescendingSecond;
//
// This converts the index into a sort code. Note that Back sort code will have left and
// right swapped.
2020-04-21 01:44:17 +00:00
//
int FrontDirectionsArray[] = {
SORTYFIRST + ASCENDINGFIRST + ASCENDINGSECOND, //"Top to bottom, left to right", // 100
SORTYFIRST + ASCENDINGFIRST + DESCENDINGSECOND, //"Top to bottom, right to left", // 101
SORTYFIRST + DESCENDINGFIRST + ASCENDINGSECOND, //"Back to Front, left to right", // 110
SORTYFIRST + DESCENDINGFIRST + DESCENDINGSECOND, //"Back to Front, right to left", // 111
SORTXFIRST + ASCENDINGFIRST + ASCENDINGSECOND, //"Left to right, Front to Back", // 000
SORTXFIRST + ASCENDINGFIRST + DESCENDINGSECOND, //"Left to right, Back to Front", // 001
SORTXFIRST + DESCENDINGFIRST + ASCENDINGSECOND, //"Right to left, Front to Back", // 010
SORTXFIRST + DESCENDINGFIRST + DESCENDINGSECOND //"Right to left, Back to Front", // 011
};
//
// Back Left/Right is opposite because it is a mirror image (coordinates are from the top)
//
int BackDirectionsArray[] = {
SORTYFIRST + ASCENDINGFIRST + DESCENDINGSECOND, //"Top to bottom, left to right", // 101
SORTYFIRST + ASCENDINGFIRST + ASCENDINGSECOND, //"Top to bottom, right to left", // 100
SORTYFIRST + DESCENDINGFIRST + DESCENDINGSECOND, //"Bottom to top, left to right", // 111
SORTYFIRST + DESCENDINGFIRST + ASCENDINGSECOND, //"Bottom to top, right to left", // 110
SORTXFIRST + DESCENDINGFIRST + ASCENDINGSECOND, //"Left to right, top to bottom", // 010
SORTXFIRST + DESCENDINGFIRST + DESCENDINGSECOND, //"Left to right, bottom to top", // 011
SORTXFIRST + ASCENDINGFIRST + ASCENDINGSECOND, //"Right to left, top to bottom", // 000
SORTXFIRST + ASCENDINGFIRST + DESCENDINGSECOND //"Right to left, bottom to top", // 001
};
#define SetSortCodes( DirArray, Code ) \
{ \
SortYFirst = ( ( DirArray[Code] & SORTYFIRST ) != 0 ); \
DescendingFirst = ( ( DirArray[Code] & DESCENDINGFIRST ) != 0 ); \
DescendingSecond = ( ( DirArray[Code] & DESCENDINGSECOND ) != 0 ); \
}
wxString AnnotateString[] = {
_( "All" ), //AnnotateAll
_( "Only front" ), //AnnotateFront
_( "Only back" ), //AnnotateBack
_( "Only selected" ) //AnnotateSelected
};
wxString ActionMessage[] = {
"", //UpdateRefDes
2020-04-21 01:44:17 +00:00
_( "Empty" ), //EmptyRefDes
_( "Invalid" ), //InvalidRefDes
_( "Excluded" ) //Exclude
};
DIALOG_BOARD_REANNOTATE::DIALOG_BOARD_REANNOTATE( PCB_EDIT_FRAME* aParentFrame )
2020-08-09 16:13:17 +00:00
: DIALOG_BOARD_REANNOTATE_BASE( aParentFrame ),
2020-11-13 02:57:11 +00:00
m_footprints( aParentFrame->GetBoard()->Footprints() )
2020-04-21 01:44:17 +00:00
{
m_Config = Kiface().KifaceSettings();
InitValues();
m_frame = aParentFrame;
m_screen = m_frame->GetScreen();
2020-11-13 02:57:11 +00:00
m_standalone = !m_frame->TestStandalone(); //Do this here forces the menu on top
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
if( m_standalone )
2020-04-21 01:44:17 +00:00
{ //Only update the schematic if not in standalone mode
m_UpdateSchematic->Enable( false );
m_UpdateSchematic->SetValue( false );
}
m_FrontRefDesStart->SetValidator( wxTextValidator( wxFILTER_DIGITS ) );
m_BackRefDesStart->SetValidator( wxTextValidator( wxFILTER_DIGITS ) );
m_sdbSizerOK->SetLabel( _( "Reannotate PCB" ) );
m_sdbSizerCancel->SetLabel( _( "Close" ) );
m_sdbSizer->Layout();
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
m_settings = aParentFrame->config();
2020-04-21 01:44:17 +00:00
wxArrayString gridslist;
2020-11-13 02:57:11 +00:00
GRID_MENU::BuildChoiceList( &gridslist, m_settings, aParentFrame );
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
if( -1 == m_gridIndex ) //If no default loaded
m_gridIndex = m_settings->m_Window.grid.last_size_idx; //Get the current grid size
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
m_sortGridx = m_frame->GetCanvas()->GetGAL()->GetGridSize().x;
m_sortGridy = m_frame->GetCanvas()->GetGAL()->GetGridSize().y;
2020-04-21 01:44:17 +00:00
m_GridChoice->Set( gridslist ); //Show the choice in the dialog
2020-11-13 02:57:11 +00:00
m_GridChoice->SetSelection( m_gridIndex );
2020-04-21 01:44:17 +00:00
for( wxRadioButton* button : m_sortButtons )
button->SetValue( false );
2020-11-13 02:57:11 +00:00
m_sortButtons[m_sortCode]->SetValue( true );
2020-04-21 01:44:17 +00:00
m_selection = m_frame->GetToolManager()->GetTool<SELECTION_TOOL>()->GetSelection();
if( !m_selection.Empty() )
2020-11-13 02:57:11 +00:00
m_annotationChoice = AnnotationChoice::AnnotateSelected;
2020-04-21 01:44:17 +00:00
for( wxRadioButton* button : AnnotateWhat )
button->SetValue( false );
2020-11-13 02:57:11 +00:00
m_annotationChoice = ( m_sortCode >= (int) AnnotateWhat.size() ) ?
AnnotationChoice::AnnotateAll :
m_annotationChoice;
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
AnnotateWhat[m_annotationChoice]->SetValue( true );
2020-04-21 01:44:17 +00:00
reannotate_down_right_bitmap->SetBitmap( KiBitmap( reannotate_right_down_xpm ) );
reannotate_right_down_bitmap->SetBitmap( KiBitmap( reannotate_left_down_xpm ) );
reannotate_down_left_bitmap->SetBitmap( KiBitmap( reannotate_right_up_xpm ) );
reannotate_left_down_bitmap->SetBitmap( KiBitmap( reannotate_left_up_xpm ) );
reannotate_up_right_bitmap->SetBitmap( KiBitmap( reannotate_down_left_xpm ) );
reannotate_right_up_bitmap->SetBitmap( KiBitmap( reannotate_up_left_xpm ) );
reannotate_up_left_bitmap->SetBitmap( KiBitmap( reannotate_down_right_xpm ) );
reannotate_left_up_bitmap->SetBitmap( KiBitmap( reannotate_up_right_xpm ) );
m_ExcludeList->SetToolTip( m_ExcludeListText->GetToolTipText() );
m_GridChoice->SetToolTip( m_SortGridText->GetToolTipText() );
// Set the reporter window filename to something sensible
2020-04-21 01:44:17 +00:00
if( m_MessageWindow->GetFileName().empty() )
{
2020-04-21 01:44:17 +00:00
wxFileName fn = m_frame->GetBoard()->GetFileName();
fn.SetName( "annotationreport" );
fn.SetExt( "txt " );
wxString fullname = fn.GetFullPath();
m_MessageWindow->SetFileName( fullname );
}
m_MessageWindow->SetPrintInfo( false ); // Suppress the "Info: " prefix
2020-11-16 11:16:44 +00:00
finishDialogSettings();
2020-04-21 01:44:17 +00:00
}
DIALOG_BOARD_REANNOTATE::~DIALOG_BOARD_REANNOTATE()
{
GetParameters(); //Get the current menu settings
PCBNEW_SETTINGS* cfg = m_frame->GetPcbNewSettings();
2020-11-13 02:57:11 +00:00
cfg->m_Reannotate.sort_on_fp_location = m_locationChoice->GetSelection() == 0;
2020-08-11 12:34:53 +00:00
cfg->m_Reannotate.remove_front_prefix = m_RemoveFrontPrefix->GetValue();
cfg->m_Reannotate.remove_back_prefix = m_RemoveBackPrefix->GetValue();
cfg->m_Reannotate.update_schematic = m_UpdateSchematic->GetValue();
cfg->m_Reannotate.exclude_locked = m_ExcludeLocked->GetValue();
2020-11-13 02:57:11 +00:00
cfg->m_Reannotate.grid_index = m_gridIndex;
cfg->m_Reannotate.sort_code = m_sortCode;
cfg->m_Reannotate.annotation_choice = m_annotationChoice;
cfg->m_Reannotate.report_severity = m_severity;
2020-08-11 12:34:53 +00:00
cfg->m_Reannotate.front_refdes_start = m_FrontRefDesStart->GetValue();
cfg->m_Reannotate.back_refdes_start = m_BackRefDesStart->GetValue();
cfg->m_Reannotate.front_prefix = m_FrontPrefix->GetValue();
cfg->m_Reannotate.back_prefix = m_BackPrefix->GetValue();
cfg->m_Reannotate.exclude_list = m_ExcludeList->GetValue();
cfg->m_Reannotate.report_file_name = m_MessageWindow->GetFileName();
2020-04-21 01:44:17 +00:00
}
2020-04-21 01:44:17 +00:00
/// Copy saved app settings to the dialog
void DIALOG_BOARD_REANNOTATE::InitValues( void )
{
PCBNEW_SETTINGS* cfg = m_frame->GetPcbNewSettings();
2020-11-13 02:57:11 +00:00
m_locationChoice->SetSelection( cfg->m_Reannotate.sort_on_fp_location ? 0 : 1 );
2020-04-21 01:44:17 +00:00
m_RemoveFrontPrefix->SetValue( cfg->m_Reannotate.remove_front_prefix );
m_RemoveBackPrefix->SetValue( cfg->m_Reannotate.remove_back_prefix );
m_UpdateSchematic->SetValue( cfg->m_Reannotate.update_schematic );
m_ExcludeLocked->SetValue( cfg->m_Reannotate.exclude_locked );
2020-11-13 02:57:11 +00:00
m_gridIndex = cfg->m_Reannotate.grid_index ;
m_sortCode = cfg->m_Reannotate.sort_code ;
m_annotationChoice = cfg->m_Reannotate.annotation_choice ;
m_severity = cfg->m_Reannotate.report_severity;
2020-04-21 01:44:17 +00:00
m_FrontRefDesStart->SetValue( cfg->m_Reannotate.front_refdes_start );
m_BackRefDesStart->SetValue( cfg->m_Reannotate.back_refdes_start );
m_FrontPrefix->SetValue( cfg->m_Reannotate.front_prefix );
m_BackPrefix->SetValue( cfg->m_Reannotate.back_prefix );
m_ExcludeList->SetValue( cfg->m_Reannotate.exclude_list );
m_MessageWindow->SetFileName( cfg->m_Reannotate.report_file_name );
}
void DIALOG_BOARD_REANNOTATE::OnCloseClick( wxCommandEvent& event )
{
EndDialog( wxID_OK );
}
//
/// Check to make sure the prefix (if there is one) is properly constructed
void DIALOG_BOARD_REANNOTATE::FilterPrefix( wxTextCtrl* aPrefix )
{
std::string tmps = VALIDPREFIX;
if( aPrefix->GetValue().empty() )
return; //Should never happen
char lastc = aPrefix->GetValue().Last();
if( isalnum( (int) lastc ) )
return;
if( std::string::npos != tmps.find( lastc ) )
return;
tmps = aPrefix->GetValue();
aPrefix->Clear();
tmps.pop_back();
aPrefix->AppendText( tmps );
}
void DIALOG_BOARD_REANNOTATE::FilterFrontPrefix( wxCommandEvent& event )
{
FilterPrefix( m_FrontPrefix );
}
void DIALOG_BOARD_REANNOTATE::FilterBackPrefix( wxCommandEvent& event )
{
FilterPrefix( m_BackPrefix );
}
void DIALOG_BOARD_REANNOTATE::OnApplyClick( wxCommandEvent& event )
{
wxString warning;
if( m_frame->GetBoard()->IsEmpty() )
{
2020-08-09 16:13:17 +00:00
ShowReport( _( "No PCB to reannotate!" ), RPT_SEVERITY_ERROR );
2020-04-21 01:44:17 +00:00
return;
}
GetParameters(); //Figure out how this is to be done
MakeSampleText( warning );
if( !IsOK( m_frame, warning ) )
return;
if( ReannotateBoard() )
ShowReport( _( "PCB and schematic successfully reannotated" ), RPT_SEVERITY_ACTION );
m_MessageWindow->SetLazyUpdate( false );
m_MessageWindow->Flush( false );
m_frame->GetCanvas()->Refresh(); //Redraw
m_frame->OnModify(); //Need to save file on exit.
}
//
/// Make the text to summarize what is about to happen
void DIALOG_BOARD_REANNOTATE::MakeSampleText( wxString& aMessage )
{
wxString tmp;
aMessage.Printf( _( "\n%s footprints will be reannotated." ),
2020-11-13 02:57:11 +00:00
_( AnnotateString[m_annotationChoice] ) );
2020-04-21 01:44:17 +00:00
if( !m_ExcludeList->GetValue().empty() )
2020-08-11 12:34:53 +00:00
{
2020-08-09 16:13:17 +00:00
aMessage += wxString::Format( _( "\nAny reference types %s will not be annotated." ),
m_ExcludeList->GetValue() );
2020-08-11 12:34:53 +00:00
}
2020-04-21 01:44:17 +00:00
if( m_ExcludeLocked->GetValue() )
aMessage += wxString::Format( _( "\nLocked footprints will not be annotated" ) );
if( !m_AnnotateBack->GetValue() )
2020-08-11 12:34:53 +00:00
{
aMessage += wxString::Format( _( "\nFront footprints will start at %s" ),
2020-08-09 16:13:17 +00:00
m_FrontRefDesStart->GetValue() );
2020-08-11 12:34:53 +00:00
}
2020-04-21 01:44:17 +00:00
if( !m_AnnotateFront->GetValue() )
2020-08-09 16:13:17 +00:00
{
2020-08-09 22:26:54 +00:00
bool frontPlusOne = ( 0 == wxAtoi( m_BackRefDesStart->GetValue() ) )
2020-08-09 16:13:17 +00:00
&& !m_AnnotateBack->GetValue();
2020-08-11 12:34:53 +00:00
aMessage += wxString::Format( _( "\nBack footprints will start at %s." ),
frontPlusOne ? _( "the last front footprint + 1" ) :
2020-08-09 16:13:17 +00:00
m_BackRefDesStart->GetValue() );
}
2020-04-21 01:44:17 +00:00
if( !m_FrontPrefix->GetValue().empty() )
{
if( m_RemoveFrontPrefix->GetValue() )
2020-08-11 12:34:53 +00:00
{
aMessage += wxString::Format( _( "\nFront footprints starting with '%s' will have "
"the prefix removed." ),
m_FrontPrefix->GetValue() );
}
2020-04-21 01:44:17 +00:00
else
2020-08-11 12:34:53 +00:00
{
aMessage += wxString::Format( _( "\nFront footprints will have '%s' inserted as a "
"prefix." ),
m_FrontPrefix->GetValue() );
}
2020-04-21 01:44:17 +00:00
}
if( !m_BackPrefix->GetValue().empty() )
{
if( m_RemoveBackPrefix->GetValue() )
2020-08-11 12:34:53 +00:00
{
aMessage += wxString::Format( _( "\nBack footprints starting with '%s' will have the "
"prefix removed." ),
m_BackPrefix->GetValue() );
}
2020-04-21 01:44:17 +00:00
else
2020-08-11 12:34:53 +00:00
{
aMessage += wxString::Format( _( "\nBack footprints will have '%s' inserted as a "
"prefix." ),
m_BackPrefix->GetValue() );
}
2020-04-21 01:44:17 +00:00
}
2020-11-13 02:57:11 +00:00
bool fpLocation = m_locationChoice->GetSelection() == 0;
2020-08-11 12:34:53 +00:00
aMessage += wxString::Format( _( "\nPrior to sorting by %s, the coordinates of which will be "
"rounded to a %s, %s grid." ),
2020-11-13 02:57:11 +00:00
fpLocation ? _( "footprint location" )
: _( "reference designator location" ),
MessageTextFromValue( m_units, m_sortGridx ),
MessageTextFromValue( m_units, m_sortGridy ) );
2020-04-21 01:44:17 +00:00
2020-08-09 16:13:17 +00:00
if( m_UpdateSchematic->GetValue() )
aMessage += _( "\nThe schematic will be updated." );
else
aMessage += _( "\nThe schematic will not be updated." );
2020-04-21 01:44:17 +00:00
ShowReport( aMessage, RPT_SEVERITY_INFO );
}
2020-08-09 16:13:17 +00:00
void DIALOG_BOARD_REANNOTATE::GetParameters()
2020-04-21 01:44:17 +00:00
{
2020-11-13 02:57:11 +00:00
m_sortCode = 0; //Convert radio button to sort direction code
2020-04-21 01:44:17 +00:00
for( wxRadioButton* sortbuttons : m_sortButtons )
{
if( sortbuttons->GetValue() )
break;
2020-11-13 02:57:11 +00:00
m_sortCode++;
2020-04-21 01:44:17 +00:00
}
2020-11-13 02:57:11 +00:00
if( m_sortCode >= (int) m_sortButtons.size() )
m_sortCode = 0;
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
m_frontPrefixString = m_FrontPrefix->GetValue();
m_backPrefixString = m_BackPrefix->GetValue();
2020-04-21 01:44:17 +00:00
//Get the chosen sort grid for rounding
2020-11-13 02:57:11 +00:00
m_gridIndex = m_GridChoice->GetSelection();
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
if( m_gridIndex >= ( int ) m_settings->m_Window.grid.sizes.size() )
2020-04-21 01:44:17 +00:00
{
2020-11-13 02:57:11 +00:00
m_sortGridx = DoubleValueFromString( EDA_UNITS::MILS,
m_settings->m_Window.grid.user_grid_x );
m_sortGridy = DoubleValueFromString( EDA_UNITS::MILS,
m_settings->m_Window.grid.user_grid_y );
2020-04-21 01:44:17 +00:00
}
else
{
2020-11-13 02:57:11 +00:00
m_sortGridx = DoubleValueFromString( EDA_UNITS::MILS,
m_settings->m_Window.grid.sizes[ m_gridIndex ] );
m_sortGridy = m_sortGridx;
2020-04-21 01:44:17 +00:00
}
int i = 0;
2020-08-09 16:13:17 +00:00
2020-04-21 01:44:17 +00:00
for( wxRadioButton* button : AnnotateWhat )
{
if( button->GetValue() )
break;
else
i++;
}
2020-11-13 02:57:11 +00:00
m_annotationChoice = ( i >= (int) AnnotateWhat.size() ) ? AnnotationChoice::AnnotateAll : i;
2020-04-21 01:44:17 +00:00
m_MessageWindow->SetLazyUpdate( true );
}
//
/// Round an int coordinate to a suitable grid
int DIALOG_BOARD_REANNOTATE::RoundToGrid( int aCoord, int aGrid )
{
if( 0 == aGrid )
aGrid = MINGRID;
int rounder;
rounder = aCoord % aGrid;
aCoord -= rounder;
if( abs( rounder ) > ( aGrid / 2 ) )
aCoord += ( aCoord < 0 ? -aGrid : aGrid );
return ( aCoord );
}
//
/// Compare function used to compare ChangeArray element for sort
/// @return true is A < B
static bool ChangeArrayCompare( const RefDesChange& aA, const RefDesChange& aB )
{
return ( aA.OldRefDesString < aB.OldRefDesString );
}
//
/// Compare function to sort footprints.
2020-04-21 01:44:17 +00:00
/// @return true if the first coordinate should be before the second coordinate
static bool ModuleCompare( const RefDesInfo& aA, const RefDesInfo& aB )
{
int X0 = aA.roundedx, X1 = aB.roundedx, Y0 = aA.roundedy, Y1 = aB.roundedy;
if( SortYFirst ) //If sorting by Y then X, swap X and Y
{
std::swap( X0, Y0 );
std::swap( X1, Y1 );
}
//If descending, same compare just swap directions
if( DescendingFirst )
std::swap( X0, X1 );
if( DescendingSecond )
std::swap( Y0, Y1 );
if( X0 < X1 )
return ( true ); //yes, its smaller
if( X0 > X1 )
return ( false ); //No its not
if( Y0 < Y1 )
return ( true ); //same but equal
return ( false );
}
//
/// Convert coordinates to wxString
/// @return the string
wxString DIALOG_BOARD_REANNOTATE::CoordTowxString( int aX, int aY )
{
2020-11-13 02:57:11 +00:00
return wxString::Format( "%s, %s",
MessageTextFromValue( m_units, aX ),
MessageTextFromValue( m_units, aY ) );
2020-04-21 01:44:17 +00:00
}
//
/// Break report into strings separated by \n and sent to the reporter
void DIALOG_BOARD_REANNOTATE::ShowReport( wxString aMessage, SEVERITY aSeverity )
{
size_t pos = 0, prev = 0;
do
{
pos = aMessage.ToStdString().find( '\n', prev );
2020-08-09 16:13:17 +00:00
m_MessageWindow->Report( aMessage.ToStdString().substr( prev, pos - prev ), aSeverity );
2020-04-21 01:44:17 +00:00
prev = pos + 1;
} while( std::string::npos != pos );
}
//
/// Create an audit trail of the changes
2020-08-09 16:13:17 +00:00
void DIALOG_BOARD_REANNOTATE::LogChangePlan()
2020-04-21 01:44:17 +00:00
{
int i = 1;
wxString message;
message.Printf( _( "\n\nThere are %i types of reference designations\n"
2020-04-21 01:44:17 +00:00
"**********************************************************\n" ),
2020-11-13 02:57:11 +00:00
(int) m_refDesTypes.size() );
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
for( RefDesTypeStr Type : m_refDesTypes ) //Show all the types of refdes
2020-04-21 01:44:17 +00:00
message += Type.RefDesType + ( 0 == ( i++ % 16 ) ? "\n" : " " );
2020-11-13 02:57:11 +00:00
if( !m_excludeArray.empty() )
2020-04-21 01:44:17 +00:00
{
wxString excludes;
2020-11-13 02:57:11 +00:00
for( wxString& exclude : m_excludeArray ) //Show the refdes we are excluding
excludes += exclude + " ";
message += wxString::Format( _( "\nExcluding: %s from reannotation\n\n" ), excludes );
2020-04-21 01:44:17 +00:00
}
message += _( "\n Change Array\n***********************\n" );
2020-11-13 02:57:11 +00:00
for( RefDesChange Change : m_changeArray )
2020-04-21 01:44:17 +00:00
{
message += wxString::Format( "%s -> %s %s %s\n", Change.OldRefDesString, Change.NewRefDes,
ActionMessage[Change.Action],
UpdateRefDes != Change.Action ? wxS( " " ) + _( "will be ignored" ) : wxString("") );
2020-04-21 01:44:17 +00:00
}
ShowReport( message, RPT_SEVERITY_INFO );
}
//
/// Create a list of the footprints and their coordinates
2020-11-18 19:50:36 +00:00
void DIALOG_BOARD_REANNOTATE::LogFootprints( const wxString& aMessage,
const std::vector<RefDesInfo>& aFootprints )
2020-04-21 01:44:17 +00:00
{
wxString message = aMessage;
2020-11-13 01:33:30 +00:00
if( aFootprints.empty() )
message += _( "\nNo footprints" );
2020-04-21 01:44:17 +00:00
else
{
int i = 1;
2020-11-13 01:33:30 +00:00
bool fpLocations = m_locationChoice->GetSelection() == 0;
2020-08-11 12:34:53 +00:00
message += wxString::Format( _( "\n*********** Sort on %s ***********" ),
2020-11-13 01:33:30 +00:00
fpLocations ? _( "Footprint Coordinates" )
: _( "Reference Designator Coordinates" ) );
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
message += wxString::Format( _( "\nSort Code %d" ), m_sortCode );
2020-04-21 01:44:17 +00:00
2020-11-13 01:33:30 +00:00
for( const RefDesInfo& mod : aFootprints )
2020-04-21 01:44:17 +00:00
{
message += wxString::Format( _( "\n%d %s Uuid: [%s], X, Y: %s, Rounded X, Y, %s" ),
2020-08-11 12:34:53 +00:00
i++,
mod.RefDesString,
mod.Uuid.AsString(),
CoordTowxString( mod.x, mod.y ),
CoordTowxString( mod.roundedx, mod.roundedy ) );
2020-04-21 01:44:17 +00:00
}
}
ShowReport( message, RPT_SEVERITY_INFO );
}
2020-04-21 01:44:17 +00:00
//
/// Actually reannotate the board
/// @return false if fail, true if success
2020-08-09 16:13:17 +00:00
bool DIALOG_BOARD_REANNOTATE::ReannotateBoard()
2020-04-21 01:44:17 +00:00
{
std::string payload;
std::vector<RefDesInfo> BadRefDes;
wxString message, badrefdes;
STRING_FORMATTER stringformatter;
RefDesChange* newref;
NETLIST netlist;
2020-11-13 11:17:15 +00:00
if( !BuildFootprintList( BadRefDes ) )
2020-04-21 01:44:17 +00:00
{
2020-08-09 16:13:17 +00:00
ShowReport( "Selected options resulted in errors! Change them and try again.",
RPT_SEVERITY_ERROR );
return false;
2020-04-21 01:44:17 +00:00
}
if( !BadRefDes.empty() )
{
message.Printf(
2020-08-11 12:34:53 +00:00
_( "\nPCB has %d empty or invalid reference designations."
"\nRecommend you run DRC with 'Test footprints against schematic' checked.\n" ),
2020-04-21 01:44:17 +00:00
(int) BadRefDes.size() );
2020-08-11 12:34:53 +00:00
for( const RefDesInfo& mod : BadRefDes )
2020-04-21 01:44:17 +00:00
{
badrefdes += wxString::Format( _( "\nRefDes: %s Footprint: %s:%s at %s on PCB." ),
2020-08-11 12:34:53 +00:00
mod.RefDesString,
mod.FPID.GetLibNickname().wx_str(),
mod.FPID.GetLibItemName().wx_str(),
CoordTowxString( mod.x, mod.y ) );
2020-04-21 01:44:17 +00:00
}
2020-08-09 16:13:17 +00:00
ShowReport( message + badrefdes + "\n", RPT_SEVERITY_WARNING );
2020-08-11 12:34:53 +00:00
message += _( "Reannotate anyway?" );
2020-04-21 01:44:17 +00:00
if( !IsOK( m_frame, message ) )
return ( false );
}
payload.clear(); //If not updating schematic no netlist error
if( m_UpdateSchematic->GetValue() )
{ //If updating schematic send a netlist
2020-11-13 15:15:52 +00:00
for( FOOTPRINT* footprint : m_footprints )
2020-04-21 01:44:17 +00:00
{ // Create a netlist
2020-11-13 02:57:11 +00:00
newref = GetNewRefDes( footprint );
2020-04-21 01:44:17 +00:00
2020-08-09 16:13:17 +00:00
if( nullptr == newref )
return false; //Not found in changelist
2020-04-21 01:44:17 +00:00
2020-08-09 16:13:17 +00:00
//add to the netlist
2020-11-13 02:57:11 +00:00
netlist.AddComponent( new COMPONENT( footprint->GetFPID(), newref->NewRefDes,
footprint->GetValue(), footprint->GetPath() ) );
2020-04-21 01:44:17 +00:00
}
netlist.Format( "pcb_netlist", &stringformatter, 0,
2020-08-09 16:13:17 +00:00
CTL_OMIT_FILTERS | CTL_OMIT_NETS | CTL_OMIT_FILTERS );
2020-04-21 01:44:17 +00:00
payload = stringformatter.GetString(); //create netlist
2020-08-09 16:13:17 +00:00
//Send netlist to eeSchema
bool attemptreannotate = m_frame->ReannotateSchematic( payload );
2020-04-21 01:44:17 +00:00
if( !attemptreannotate )
{ //Didn't get a valid reply
ShowReport( _( "\nReannotate failed!\n" ), RPT_SEVERITY_WARNING );
2020-08-09 16:13:17 +00:00
return false;
2020-04-21 01:44:17 +00:00
}
} //If updating schematic
2020-11-13 11:17:15 +00:00
bool reannotateOk = payload.size( ) == 0;
2020-04-21 01:44:17 +00:00
2020-11-13 11:17:15 +00:00
ShowReport( payload, reannotateOk ? RPT_SEVERITY_ACTION : RPT_SEVERITY_ERROR );
2020-04-21 01:44:17 +00:00
BOARD_COMMIT commit( m_frame );
2020-11-13 11:17:15 +00:00
if( reannotateOk )//Only update if no errors
{
2020-04-21 01:44:17 +00:00
2020-11-13 15:15:52 +00:00
for( FOOTPRINT* footprint : m_footprints )
2020-11-13 11:17:15 +00:00
{
newref = GetNewRefDes( footprint );
2020-04-21 01:44:17 +00:00
2020-08-09 16:13:17 +00:00
if( nullptr == newref )
return false;
2020-04-21 01:44:17 +00:00
2020-11-13 11:17:15 +00:00
commit.Modify( footprint ); // Make a copy for undo
footprint->SetReference( newref->NewRefDes ); // Update the PCB reference
m_frame->GetCanvas()->GetView()->Update( footprint ); // Touch the footprint
2020-04-21 01:44:17 +00:00
}
}
commit.Push( "Geographic reannotation" );
2020-11-13 11:17:15 +00:00
return reannotateOk;
2020-04-21 01:44:17 +00:00
}
//
2020-11-13 11:17:15 +00:00
/// Build the footprint lists, sort it, filter for excludes, then build the change list
2020-04-21 01:44:17 +00:00
/// @returns true if success, false if errors
2020-11-13 11:17:15 +00:00
bool DIALOG_BOARD_REANNOTATE::BuildFootprintList( std::vector<RefDesInfo>& aBadRefDes )
2020-04-21 01:44:17 +00:00
{
2020-11-13 11:17:15 +00:00
bool annotateSelected;
bool annotateFront = m_AnnotateFront->GetValue(); //Unless only doing back
bool annotateBack = m_AnnotateBack->GetValue(); //Unless only doing front
bool skipLocked = m_ExcludeLocked->GetValue();
2020-04-21 01:44:17 +00:00
int errorcount = 0;
unsigned int backstartrefdes;
2020-11-13 02:57:11 +00:00
size_t firstnum = 0;
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
m_frontFootprints.clear();
m_backFootprints.clear();
m_excludeArray.clear();
m_footprints = m_frame->GetBoard()->Footprints();
2020-04-21 01:44:17 +00:00
std::vector<KIID> selected;
if( m_AnnotateSelection->GetValue() )
{
for( EDA_ITEM* item : m_selection )
2020-11-13 02:57:11 +00:00
{
//Get the timestamps of selected footprints
2020-11-13 12:21:02 +00:00
if( item->Type() == PCB_FOOTPRINT_T )
2020-04-21 01:44:17 +00:00
selected.push_back( item->m_Uuid );
}
}
2020-11-13 11:17:15 +00:00
annotateSelected = !selected.empty();
2020-04-21 01:44:17 +00:00
wxString exclude;
for( auto thischar : m_ExcludeList->GetValue() )
{ //Break exclude list into words
if( ( ' ' == thischar ) || ( ',' == thischar ) )
{
2020-11-13 02:57:11 +00:00
m_excludeArray.push_back( exclude );
2020-04-21 01:44:17 +00:00
exclude.clear();
}
else
exclude += thischar;
if( !exclude.empty() )
2020-11-13 02:57:11 +00:00
m_excludeArray.push_back( exclude );
2020-04-21 01:44:17 +00:00
}
2020-11-13 11:17:15 +00:00
RefDesInfo fpData;
2020-08-11 12:34:53 +00:00
bool useModuleLocation = m_locationChoice->GetSelection() == 0;
2020-04-21 01:44:17 +00:00
2020-11-13 15:15:52 +00:00
for( FOOTPRINT* footprint : m_footprints )
2020-04-21 01:44:17 +00:00
{
2020-11-13 11:17:15 +00:00
fpData.Uuid = footprint->m_Uuid;
fpData.RefDesString = footprint->GetReference();
fpData.FPID = footprint->GetFPID();
fpData.x = useModuleLocation ? footprint->GetPosition().x
: footprint->Reference().GetPosition().x;
fpData.y = useModuleLocation ? footprint->GetPosition().y
: footprint->Reference().GetPosition().y;
fpData.roundedx = RoundToGrid( fpData.x, m_sortGridx ); //Round to sort
fpData.roundedy = RoundToGrid( fpData.y, m_sortGridy );
fpData.Front = footprint->GetLayer() == F_Cu;
fpData.Action = UpdateRefDes; //Usually good
if( fpData.RefDesString.IsEmpty() )
{
fpData.Action = EmptyRefDes;
}
2020-04-21 01:44:17 +00:00
else
{
2020-11-13 11:17:15 +00:00
firstnum = fpData.RefDesString.find_first_of( "0123456789" );
2020-04-21 01:44:17 +00:00
if( std::string::npos == firstnum )
2020-11-13 11:17:15 +00:00
fpData.Action = InvalidRefDes; //do not change ref des such as 12 or +1, or L
2020-04-21 01:44:17 +00:00
}
2020-08-09 16:13:17 +00:00
//Get the type (R, C, etc)
2020-11-13 11:17:15 +00:00
fpData.RefDesType = fpData.RefDesString.substr( 0, firstnum );
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
for( wxString excluded : m_excludeArray )
2020-04-21 01:44:17 +00:00
{
2020-11-13 11:17:15 +00:00
if( excluded == fpData.RefDesType ) //Am I supposed to exclude this type?
2020-04-21 01:44:17 +00:00
{
2020-11-13 11:17:15 +00:00
fpData.Action = Exclude; //Yes
2020-04-21 01:44:17 +00:00
break;
}
}
if(( fpData.Front && annotateBack ) || // If a front fp and doing backs only
( !fpData.Front && annotateFront ) || // If a back fp and doing front only
2020-11-13 11:17:15 +00:00
( footprint->IsLocked() && skipLocked ) ) // If excluding locked and it is locked
2020-11-13 02:57:11 +00:00
{
2020-11-13 11:17:15 +00:00
fpData.Action = Exclude;
2020-11-13 02:57:11 +00:00
}
2020-04-21 01:44:17 +00:00
2020-11-13 11:17:15 +00:00
if( annotateSelected )
{ // If onnly annotating selected c
fpData.Action = Exclude; // Assume it isn't selected
2020-04-21 01:44:17 +00:00
for( KIID sel : selected )
{
2020-11-13 11:17:15 +00:00
if( fpData.Uuid == sel )
{ // Found in selected footprints
fpData.Action = UpdateRefDes; // Update it
2020-04-21 01:44:17 +00:00
break;
}
}
}
2020-11-13 11:17:15 +00:00
if( fpData.Front )
m_frontFootprints.push_back( fpData );
2020-04-21 01:44:17 +00:00
else
2020-11-13 11:17:15 +00:00
m_backFootprints.push_back( fpData );
2020-04-21 01:44:17 +00:00
}
2020-11-13 02:57:11 +00:00
SetSortCodes( FrontDirectionsArray, m_sortCode ); //Determine the sort order for the front
sort( m_frontFootprints.begin(), m_frontFootprints.end(), ModuleCompare ); //Sort the front footprints
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
SetSortCodes( BackDirectionsArray, m_sortCode ); //Determine the sort order for the back
sort( m_backFootprints.begin(), m_backFootprints.end(), ModuleCompare ); //Sort the back footprints
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
m_refDesTypes.clear();
m_changeArray.clear();
2020-08-09 22:26:54 +00:00
backstartrefdes = wxAtoi( m_BackRefDesStart->GetValue() );
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
if( !m_frontFootprints.empty() )
{
BuildChangeArray( m_frontFootprints, wxAtoi( m_FrontRefDesStart->GetValue() ),
2020-08-09 16:13:17 +00:00
m_FrontPrefix->GetValue(), m_RemoveFrontPrefix->GetValue(), aBadRefDes );
2020-11-13 02:57:11 +00:00
}
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
if( !m_backFootprints.empty() )
{
BuildChangeArray( m_backFootprints, backstartrefdes, m_BackPrefix->GetValue(),
2020-08-09 16:13:17 +00:00
m_RemoveBackPrefix->GetValue(), aBadRefDes );
2020-11-13 02:57:11 +00:00
}
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
if( !m_changeArray.empty() )
sort( m_changeArray.begin(), m_changeArray.end(), ChangeArrayCompare );
2020-04-21 01:44:17 +00:00
LogChangePlan();
2020-11-13 02:57:11 +00:00
size_t changearraysize = m_changeArray.size();
2020-08-09 16:13:17 +00:00
2020-04-21 01:44:17 +00:00
for( size_t i = 0; i < changearraysize; i++ ) //Scan through for duplicates if update or skip
{
2020-11-13 02:57:11 +00:00
if( ( m_changeArray[i].Action != EmptyRefDes )
&& ( m_changeArray[i].Action != InvalidRefDes ) )
2020-04-21 01:44:17 +00:00
{
for( size_t j = i + 1; j < changearraysize; j++ )
{
2020-11-13 02:57:11 +00:00
if( m_changeArray[i].NewRefDes == m_changeArray[j].NewRefDes )
2020-04-21 01:44:17 +00:00
{
2020-11-13 02:57:11 +00:00
ShowReport( "Duplicate instances of " + m_changeArray[j].NewRefDes,
2020-08-09 16:13:17 +00:00
RPT_SEVERITY_ERROR );
2020-04-21 01:44:17 +00:00
if( errorcount++ > MAXERROR )
{
ShowReport( _( "Aborted: too many errors" ), RPT_SEVERITY_ERROR );
2020-04-21 01:44:17 +00:00
break;
}
}
}
}
if( errorcount > MAXERROR )
break;
}
return ( 0 == errorcount );
2020-08-09 16:13:17 +00:00
}
2020-04-21 01:44:17 +00:00
//
2020-11-13 11:17:15 +00:00
/// Scan through the footprint arrays and create the from -> to array
2020-11-13 01:33:30 +00:00
void DIALOG_BOARD_REANNOTATE::BuildChangeArray( std::vector<RefDesInfo>& aFootprints,
2020-08-09 16:13:17 +00:00
unsigned int aStartRefDes, wxString aPrefix,
bool aRemovePrefix,
std::vector<RefDesInfo>& aBadRefDes )
2020-04-21 01:44:17 +00:00
{
size_t i;
RefDesChange change;
RefDesTypeStr newtype;
wxString refdestype;
size_t prefixsize = aPrefix.size();
bool haveprefix = ( 0 != prefixsize ); //Do I have a prefix?
bool addprefix = haveprefix & !aRemovePrefix; //Yes- and I'm not removing it
aRemovePrefix &= haveprefix; //Only remove if I have a prefix
bool prefixpresent; //Prefix found
2020-11-13 01:33:30 +00:00
wxString logstring = ( aFootprints.front().Front ) ? _( "\n\nFront Footprints" )
: _( "\n\nBack Footprints" );
LogFootprints( logstring, aFootprints );
2020-04-21 01:44:17 +00:00
if( 0 != aStartRefDes ) //Initialize the change array if present
2020-11-13 01:33:30 +00:00
{
2020-11-13 02:57:11 +00:00
for( i = 0; i < m_refDesTypes.size(); i++ )
m_refDesTypes[i].RefDesCount = aStartRefDes;
2020-11-13 01:33:30 +00:00
}
2020-04-21 01:44:17 +00:00
2020-11-13 11:17:15 +00:00
for( RefDesInfo fpData : aFootprints )
{
change.Uuid = fpData.Uuid;
change.Action = fpData.Action;
change.OldRefDesString = fpData.RefDesString;
change.NewRefDes = fpData.RefDesString;
change.Front = fpData.Front;
2020-04-21 01:44:17 +00:00
2020-11-13 11:17:15 +00:00
if( fpData.RefDesString.IsEmpty() )
fpData.Action = EmptyRefDes;
2020-04-21 01:44:17 +00:00
if( ( change.Action == EmptyRefDes ) || ( change.Action == InvalidRefDes ) )
{
2020-11-13 02:57:11 +00:00
m_changeArray.push_back( change );
2020-11-13 11:17:15 +00:00
aBadRefDes.push_back( fpData );
2020-04-21 01:44:17 +00:00
continue;
}
if( change.Action == UpdateRefDes )
{
2020-11-13 11:17:15 +00:00
refdestype = fpData.RefDesType;
prefixpresent = ( 0 == fpData.RefDesType.find( aPrefix ) );
2020-04-21 01:44:17 +00:00
if( addprefix && !prefixpresent )
2020-11-13 11:17:15 +00:00
fpData.RefDesType.insert( 0, aPrefix ); //Add prefix once only
2020-04-21 01:44:17 +00:00
if( aRemovePrefix && prefixpresent ) //If there is a prefix remove it
2020-11-13 11:17:15 +00:00
fpData.RefDesType.erase( 0, prefixsize );
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
for( i = 0; i < m_refDesTypes.size(); i++ ) //See if it is in the types array
{
2020-11-13 11:17:15 +00:00
if( m_refDesTypes[i].RefDesType == fpData.RefDesType ) //Found it!
2020-04-21 01:44:17 +00:00
break;
2020-11-13 02:57:11 +00:00
}
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
if( i == m_refDesTypes.size() )
2020-04-21 01:44:17 +00:00
{ //Wasn't in the types array so add it
2020-11-13 11:17:15 +00:00
newtype.RefDesType = fpData.RefDesType;
2020-08-09 16:13:17 +00:00
newtype.RefDesCount = ( aStartRefDes == 0 ? 1 : aStartRefDes );
2020-11-13 02:57:11 +00:00
m_refDesTypes.push_back( newtype );
2020-04-21 01:44:17 +00:00
}
2020-11-13 02:57:11 +00:00
change.NewRefDes = m_refDesTypes[i].RefDesType
+ std::to_string( m_refDesTypes[i].RefDesCount++ );
2020-04-21 01:44:17 +00:00
}
2020-11-13 02:57:11 +00:00
m_changeArray.push_back( change ); //Add to the change array
2020-08-09 16:13:17 +00:00
}
}
2020-04-21 01:44:17 +00:00
2020-04-21 01:44:17 +00:00
//
2020-11-13 02:57:11 +00:00
/// @returns the new refdes for this footprint
2020-11-13 15:15:52 +00:00
RefDesChange* DIALOG_BOARD_REANNOTATE::GetNewRefDes( FOOTPRINT* aFootprint )
2020-04-21 01:44:17 +00:00
{
size_t i;
2020-11-13 02:57:11 +00:00
for( i = 0; i < m_changeArray.size(); i++ )
{
if( aFootprint->m_Uuid == m_changeArray[i].Uuid )
return ( &m_changeArray[i] );
}
2020-04-21 01:44:17 +00:00
2020-11-13 02:57:11 +00:00
ShowReport( _( "Footprint not found in changelist" ) + wxS( " " ) + aFootprint->GetReference(),
RPT_SEVERITY_ERROR );
2020-08-09 16:13:17 +00:00
return nullptr; //Should never happen
2020-04-21 01:44:17 +00:00
}