kicad-source/eeschema/dialogs/dialog_change_symbols.cpp
2025-08-09 21:52:08 +01:00

892 lines
30 KiB
C++

/*
* This program source code file is part of KiCad, a free EDA CAD application.
*
* Copyright (C) 2020-2021 CERN
* Copyright The KiCad Developers, see AUTHORS.txt for contributors.
*
* @author Wayne Stambaugh <stambaughw@gmail.com>
*
* This program is free software: you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation, either version 3 of the License, or (at your
* option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <algorithm>
#include <bitmaps.h>
#include <connection_graph.h>
#include <string_utils.h> // WildCompareString
#include <kiway.h>
#include <refdes_utils.h>
#include <core/kicad_algo.h>
#include <dialog_change_symbols.h>
#include <sch_symbol.h>
#include <sch_edit_frame.h>
#include <sch_screen.h>
#include <schematic.h>
#include <template_fieldnames.h>
#include <widgets/wx_html_report_panel.h>
#include <widgets/std_bitmap_button.h>
#include <sch_commit.h>
bool g_selectRefDes = false;
bool g_selectValue = false;
DIALOG_CHANGE_SYMBOLS::DIALOG_CHANGE_SYMBOLS( SCH_EDIT_FRAME* aParent, SCH_SYMBOL* aSymbol, MODE aMode ) :
DIALOG_CHANGE_SYMBOLS_BASE( aParent ),
m_symbol( aSymbol),
m_mode( aMode )
{
wxASSERT( aParent );
if( m_mode == MODE::UPDATE )
{
m_newIdSizer->Show( false );
}
else
{
m_matchAll->SetLabel( _( "Change all symbols in schematic" ) );
SetTitle( _( "Change Symbols" ) );
m_matchSizer->FindItem( m_matchAll )->Show( false );
m_matchByReference->SetLabel( _( "Change symbols matching reference designator:" ) );
m_matchByValue->SetLabel( _( "Change symbols matching value:" ) );
m_matchById->SetLabel( _( "Change symbols matching library identifier:" ) );
m_updateFieldsSizer->GetStaticBox()->SetLabel( _( "Update Fields" ) );
m_removeExtraBox->SetLabel( _( "Remove fields if not in new symbol" ) );
m_resetEmptyFields->SetLabel( _( "Reset fields if empty in new symbol" ) );
m_resetFieldText->SetLabel( _( "Update field text" ) );
m_resetFieldVisibilities->SetLabel( _( "Update field visibilities" ) );
m_resetFieldEffects->SetLabel( _( "Update field sizes and styles" ) );
m_resetFieldPositions->SetLabel( _( "Update field positions" ) );
m_resetAttributes->SetLabel( _( "Update symbol attributes" ) );
m_resetPinTextVisibility->SetLabel( _( "Update pin name/number visibilities" ) );
m_resetAlternatePin->SetLabel( _( "Reset alternate pin functions" ) );
m_resetCustomPower->SetLabel( _( "Reset custom power symbols" ) );
if( m_symbol )
m_matchBySelection->SetLabel( _( "Change selected symbol(s)" ) );
}
if( !m_symbol )
m_matchSizer->FindItem( m_matchBySelection )->Show( false );
m_matchIdBrowserButton->SetBitmap( KiBitmapBundle( BITMAPS::small_library ) );
m_newIdBrowserButton->SetBitmap( KiBitmapBundle( BITMAPS::small_library ) );
m_matchSizer->SetEmptyCellSize( wxSize( 0, 0 ) );
m_matchSizer->Layout();
for( FIELD_T fieldId : MANDATORY_FIELDS )
{
int listIdx = (int) m_fieldsBox->GetCount();
m_fieldsBox->Append( GetDefaultFieldName( fieldId, DO_TRANSLATE ) );
// List boxes aren't currently handled in DIALOG_SHIM's control-state-save/restore
if( fieldId == FIELD_T::REFERENCE )
m_fieldsBox->Check( listIdx, g_selectRefDes );
else if( fieldId == FIELD_T::VALUE )
m_fieldsBox->Check( listIdx, g_selectValue );
else
m_fieldsBox->Check( listIdx, true );
m_mandatoryFieldListIndexes[fieldId] = listIdx;
}
updateFieldsList();
// initialize controls based on m_mode in case there is no saved state yet
m_removeExtraBox->SetValue( m_mode == MODE::CHANGE ? false : false );
m_resetEmptyFields->SetValue( m_mode == MODE::CHANGE ? false : false );
m_resetFieldText->SetValue( m_mode == MODE::CHANGE ? true : true );
m_resetFieldVisibilities->SetValue( m_mode == MODE::CHANGE ? true : false );
m_resetFieldEffects->SetValue( m_mode == MODE::CHANGE ? true : false );
m_resetFieldPositions->SetValue( m_mode == MODE::CHANGE ? true : false );
m_resetAttributes->SetValue( m_mode == MODE::CHANGE ? true : false );
m_resetPinTextVisibility->SetValue( m_mode == MODE::CHANGE ? true : false );
m_resetCustomPower->SetValue( m_mode == MODE::CHANGE ? false : false );
m_resetAlternatePin->SetValue( m_mode == MODE::CHANGE ? true : false );
m_messagePanel->SetLazyUpdate( true );
m_messagePanel->SetFileName( Prj().GetProjectPath() + wxT( "report.txt" ) );
// DIALOG_SHIM needs a unique hash_key because classname is not sufficient
// because the update and change versions of this dialog have different controls.
m_hash_key = TO_UTF8( GetTitle() );
wxString okLabel = m_mode == MODE::CHANGE ? _( "Change" ) : _( "Update" );
SetupStandardButtons( { { wxID_OK, okLabel },
{ wxID_CANCEL, _( "Close" ) } } );
// Now all widgets have the size fixed, call FinishDialogSettings
finishDialogSettings();
}
bool DIALOG_CHANGE_SYMBOLS::TransferDataToWindow()
{
if( m_symbol )
{
SCH_SHEET_PATH* currentSheet = &m_symbol->Schematic()->CurrentSheet();
m_specifiedReference->ChangeValue( m_symbol->GetRef( currentSheet ) );
m_specifiedValue->ChangeValue( UnescapeString( m_symbol->GetField( FIELD_T::VALUE )->GetText() ) );
m_specifiedId->ChangeValue( UnescapeString( m_symbol->GetLibId().Format() ) );
}
if( m_symbol && m_symbol->IsSelected() )
m_matchBySelection->SetValue( true );
else if( m_mode == MODE::UPDATE )
m_matchAll->SetValue( true );
else
m_matchByReference->SetValue( true );
return true;
}
void DIALOG_CHANGE_SYMBOLS::onMatchByAll( wxCommandEvent& aEvent )
{
updateFieldsList();
}
void DIALOG_CHANGE_SYMBOLS::onMatchBySelected( wxCommandEvent& aEvent )
{
updateFieldsList();
}
void DIALOG_CHANGE_SYMBOLS::onMatchByReference( wxCommandEvent& aEvent )
{
updateFieldsList();
m_specifiedReference->SetFocus();
}
void DIALOG_CHANGE_SYMBOLS::onMatchByValue( wxCommandEvent& aEvent )
{
updateFieldsList();
m_specifiedValue->SetFocus();
}
void DIALOG_CHANGE_SYMBOLS::onMatchById( wxCommandEvent& aEvent )
{
updateFieldsList();
m_specifiedId->SetFocus();
}
void DIALOG_CHANGE_SYMBOLS::onMatchTextKillFocus( wxFocusEvent& event )
{
updateFieldsList();
event.Skip(); // Mandatory in wxFocusEvent
}
void DIALOG_CHANGE_SYMBOLS::onMatchIDKillFocus( wxFocusEvent& event )
{
updateFieldsList();
event.Skip(); // Mandatory in wxFocusEvent
}
void DIALOG_CHANGE_SYMBOLS::onNewLibIDKillFocus( wxFocusEvent& event )
{
updateFieldsList();
event.Skip(); // Mandatory in wxFocusEvent
}
DIALOG_CHANGE_SYMBOLS::~DIALOG_CHANGE_SYMBOLS()
{
// List boxes aren't currently handled in DIALOG_SHIM's control-state-save/restore
g_selectRefDes = m_fieldsBox->IsChecked( m_mandatoryFieldListIndexes[FIELD_T::REFERENCE] );
g_selectValue = m_fieldsBox->IsChecked( m_mandatoryFieldListIndexes[FIELD_T::VALUE] );
}
wxString getLibIdValue( const wxTextCtrl* aCtrl )
{
wxString rawValue = aCtrl->GetValue();
wxString itemName;
wxString libName = rawValue.BeforeFirst( ':', &itemName );
return EscapeString( libName, CTX_LIBID ) + ':' + EscapeString( itemName, CTX_LIBID );
}
void DIALOG_CHANGE_SYMBOLS::launchMatchIdSymbolBrowser( wxCommandEvent& aEvent )
{
wxString newName = getLibIdValue( m_specifiedId );
if( KIWAY_PLAYER* frame = Kiway().Player( FRAME_SYMBOL_CHOOSER, true, this ) )
{
if( frame->ShowModal( &newName, this ) )
{
m_specifiedId->SetValue( UnescapeString( newName ) );
updateFieldsList();
}
frame->Destroy();
}
}
void DIALOG_CHANGE_SYMBOLS::launchNewIdSymbolBrowser( wxCommandEvent& aEvent )
{
wxString newName = getLibIdValue( m_newId );
if( KIWAY_PLAYER* frame = Kiway().Player( FRAME_SYMBOL_CHOOSER, true, this ) )
{
if( frame->ShowModal( &newName, this ) )
{
m_newId->SetValue( UnescapeString( newName ) );
updateFieldsList();
}
frame->Destroy();
}
}
void DIALOG_CHANGE_SYMBOLS::updateFieldsList()
{
SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
wxCHECK( frame, /* void */ );
// Load non-mandatory fields from all matching symbols and their library symbols
std::set<wxString> fieldNames;
for( SCH_SHEET_PATH& instance : frame->Schematic().Hierarchy() )
{
SCH_SCREEN* screen = instance.LastScreen();
wxCHECK2( screen, continue );
for( SCH_ITEM* item : screen->Items().OfType( SCH_SYMBOL_T ) )
{
SCH_SYMBOL* symbol = dynamic_cast<SCH_SYMBOL*>( item );
wxCHECK2( symbol, continue );
if( !isMatch( symbol, &instance ) )
continue;
for( SCH_FIELD& field : symbol->GetFields() )
{
if( !field.IsMandatory() && !field.IsPrivate() )
fieldNames.insert( field.GetName() );
}
if( m_mode == MODE::UPDATE && symbol->GetLibId().IsValid() )
{
LIB_SYMBOL* libSymbol = frame->GetLibSymbol( symbol->GetLibId() );
if( libSymbol )
{
std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
std::vector<SCH_FIELD*> orderedLibFields;
flattenedSymbol->GetFields( orderedLibFields );
for( SCH_FIELD* libField : orderedLibFields )
{
if( !libField->IsMandatory() && !libField->IsPrivate() )
fieldNames.insert( libField->GetName() );
}
}
}
}
}
// Load non-mandatory fields from the change-to library symbol
if( m_mode == MODE::CHANGE )
{
LIB_ID newId;
newId.Parse( getLibIdValue( m_newId ) );
if( newId.IsValid() )
{
LIB_SYMBOL* libSymbol = frame->GetLibSymbol( newId );
if( libSymbol )
{
std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
std::vector<SCH_FIELD*> orderedLibFields;
flattenedSymbol->GetFields( orderedLibFields );
for( SCH_FIELD* libField : orderedLibFields )
{
if( !libField->IsMandatory() && !libField->IsPrivate() )
fieldNames.insert( libField->GetName() );
}
}
}
}
// Update the listbox widget
wxArrayInt checkedItems;
wxArrayString checkedNames;
m_fieldsBox->GetCheckedItems( checkedItems );
for( int ii : checkedItems )
checkedNames.push_back( m_fieldsBox->GetString( ii ) );
bool allChecked = true;
for( int ii = 0; ii < (int) m_fieldsBox->GetCount(); ++ii )
{
if( !m_fieldsBox->IsChecked( ii )
&& ii != m_mandatoryFieldListIndexes[FIELD_T::REFERENCE]
&& ii != m_mandatoryFieldListIndexes[FIELD_T::VALUE] )
{
allChecked = false;
}
}
auto isMandatoryField =
[&]( int listbox_idx )
{
for( FIELD_T fieldId : MANDATORY_FIELDS )
{
if( m_mandatoryFieldListIndexes[fieldId] == listbox_idx )
return true;
}
return false;
};
for( int ii = (int) m_fieldsBox->GetCount() - 1; ii >= 0; --ii )
{
if( isMandatoryField( ii ) )
continue;
m_fieldsBox->Delete( ii );
}
for( const wxString& fieldName : fieldNames )
{
m_fieldsBox->Append( fieldName );
if( allChecked || alg::contains( checkedNames, fieldName ) )
m_fieldsBox->Check( m_fieldsBox->GetCount() - 1, true );
}
}
void DIALOG_CHANGE_SYMBOLS::selectAll( bool aSelect )
{
for( unsigned i = 0; i < m_fieldsBox->GetCount(); ++i )
m_fieldsBox->Check( i, aSelect );
}
void DIALOG_CHANGE_SYMBOLS::checkAll( bool aCheck )
{
m_removeExtraBox->SetValue( aCheck );
m_resetEmptyFields->SetValue( aCheck );
m_resetFieldText->SetValue( aCheck );
m_resetFieldVisibilities->SetValue( aCheck );
m_resetFieldEffects->SetValue( aCheck );
m_resetFieldPositions->SetValue( aCheck );
m_resetPinTextVisibility->SetValue( aCheck );
m_resetAlternatePin->SetValue( aCheck );
m_resetAttributes->SetValue( aCheck );
m_resetCustomPower->SetValue( aCheck );
}
void DIALOG_CHANGE_SYMBOLS::onOkButtonClicked( wxCommandEvent& aEvent )
{
SCH_EDIT_FRAME* parent = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
wxCHECK( parent, /* void */ );
wxBusyCursor dummy;
SCH_COMMIT commit( parent );
m_messagePanel->Clear();
m_messagePanel->Flush( false );
// Create the set of fields to be updated. Use non translated (canonical) names
// for mandatory fields
m_updateFields.clear();
for( unsigned ii = 0; ii < m_fieldsBox->GetCount(); ++ii )
{
if( m_fieldsBox->IsChecked( ii ) )
m_updateFields.insert( m_fieldsBox->GetString( ii ) );
}
for( FIELD_T fieldId : MANDATORY_FIELDS )
{
if( m_fieldsBox->IsChecked( m_mandatoryFieldListIndexes[fieldId] ) )
m_updateFields.insert( GetCanonicalFieldName( fieldId ) );
}
if( processMatchingSymbols( &commit) )
commit.Push( m_mode == MODE::CHANGE ? _( "Change Symbols" ) : _( "Update Symbols" ) );
m_messagePanel->Flush( false );
}
bool DIALOG_CHANGE_SYMBOLS::isMatch( SCH_SYMBOL* aSymbol, SCH_SHEET_PATH* aInstance )
{
SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
wxCHECK( frame, false );
if( !aSymbol )
return false;
if( m_matchAll->GetValue() )
{
return true;
}
else if( m_matchBySelection->GetValue() )
{
return aSymbol == m_symbol || aSymbol->IsSelected();
}
else if( m_matchByReference->GetValue() )
{
return WildCompareString( m_specifiedReference->GetValue(),
UnescapeString( aSymbol->GetRef( aInstance, false ) ),
false );
}
else if( m_matchByValue->GetValue() )
{
return WildCompareString( m_specifiedValue->GetValue(),
UnescapeString( aSymbol->GetField( FIELD_T::VALUE )->GetText() ),
false );
}
else if( m_matchById )
{
LIB_ID id;
id.Parse( getLibIdValue( m_specifiedId ) );
return aSymbol->GetLibId() == id;
}
return false;
}
int DIALOG_CHANGE_SYMBOLS::processMatchingSymbols( SCH_COMMIT* aCommit )
{
SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
wxCHECK( frame, false );
LIB_ID newId;
int matchesProcessed = 0;
SCH_SYMBOL* symbol = nullptr;
if( m_mode == MODE::CHANGE )
{
newId.Parse( getLibIdValue( m_newId ) );
if( !newId.IsValid() )
return false;
}
std::map<SCH_SYMBOL*, SYMBOL_CHANGE_INFO> symbols;
for( SCH_SHEET_PATH& instance : frame->Schematic().Hierarchy() )
{
SCH_SCREEN* screen = instance.LastScreen();
wxCHECK2( screen, continue );
// Fetch all the symbols that meet the change criteria.
for( SCH_ITEM* item : screen->Items().OfType( SCH_SYMBOL_T ) )
{
symbol = static_cast<SCH_SYMBOL*>( item );
wxCHECK2( symbol, continue );
if( !isMatch( symbol, &instance ) )
continue;
if( m_mode == MODE::UPDATE )
newId = symbol->GetLibId();
auto it = symbols.find( symbol );
if( it == symbols.end() )
{
SYMBOL_CHANGE_INFO info;
info.m_Instances.emplace_back( instance );
info.m_LibId = newId;
symbols.insert( { symbol, info } );
}
else
{
it->second.m_Instances.emplace_back( instance );
}
}
}
if( symbols.size() > 0 )
matchesProcessed += processSymbols( aCommit, symbols );
else
m_messagePanel->Report( _( "*** No symbols matching criteria found ***" ), RPT_SEVERITY_ERROR );
frame->GetCurrentSheet().UpdateAllScreenReferences();
return matchesProcessed;
}
int DIALOG_CHANGE_SYMBOLS::processSymbols( SCH_COMMIT* aCommit, const std::map<SCH_SYMBOL*,
SYMBOL_CHANGE_INFO>& aSymbols )
{
wxCHECK( !aSymbols.empty(), 0 );
int matchesProcessed = 0;
SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
wxString msg;
wxCHECK( frame, 0 );
std::map<SCH_SYMBOL*, SYMBOL_CHANGE_INFO> symbols = aSymbols;
std::map<SCH_SYMBOL*, SYMBOL_CHANGE_INFO>::iterator it = symbols.begin();
// Remove all symbols that don't have a valid library symbol link or enough units to
// satisfy the library symbol update.
while( it != symbols.end() )
{
SCH_SYMBOL* symbol = it->first;
wxCHECK2( symbol && it->second.m_LibId.IsValid(), continue );
LIB_SYMBOL* libSymbol = frame->GetLibSymbol( it->second.m_LibId );
if( !libSymbol )
{
msg = getSymbolReferences( *symbol, it->second.m_LibId );
msg << wxT( ": " ) << _( "*** symbol not found ***" );
m_messagePanel->Report( msg, RPT_SEVERITY_ERROR );
it = symbols.erase( it );
continue;
}
std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
if( flattenedSymbol->GetUnitCount() < symbol->GetUnit() )
{
msg = getSymbolReferences( *symbol, it->second.m_LibId );
msg << wxT( ": " ) << _( "*** new symbol has too few units ***" );
m_messagePanel->Report( msg, RPT_SEVERITY_ERROR );
it = symbols.erase( it );
}
else
{
++it;
}
}
// Removing the symbol needs to be done before the LIB_SYMBOL is changed to prevent stale
// library symbols in the schematic file.
for( const auto& [ symbol, symbol_change_info ] : symbols )
{
wxCHECK( symbol && !symbol_change_info.m_Instances.empty(), 0 );
SCH_SCREEN* screen = symbol_change_info.m_Instances[0].LastScreen();
wxCHECK( screen, 0 );
screen->Remove( symbol );
SCH_SYMBOL* symbol_copy = static_cast<SCH_SYMBOL*>( symbol->Clone() );
aCommit->Modified( symbol, symbol_copy, screen );
CONNECTION_GRAPH* connectionGraph = screen->Schematic()->ConnectionGraph();
// When we replace the lib symbol below, we free the associated pins if the new symbol has
// fewer than the original. This will cause the connection graph to be out of date unless
// we replace references in the graph to the old symbol/pins with references to the ones
// stored in the undo stack.
if( connectionGraph )
connectionGraph->ExchangeItem( symbol, symbol_copy );
}
for( const auto& [ symbol, symbol_change_info ] : symbols )
{
// Remember initial link before changing for diags purpose
wxString initialLibLinkName = UnescapeString( symbol->GetLibId().Format() );
if( symbol_change_info.m_LibId != symbol->GetLibId() )
symbol->SetLibId( symbol_change_info.m_LibId );
LIB_SYMBOL* libSymbol = frame->GetLibSymbol( symbol_change_info.m_LibId );
std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
SCH_SCREEN* screen = symbol_change_info.m_Instances[0].LastScreen();
symbol->SetLibSymbol( flattenedSymbol.release() );
if( m_resetAttributes->GetValue() )
{
// Fetch the attributes from the *flattened* library symbol. They are not supported
// in derived symbols.
symbol->SetExcludedFromSim( symbol->GetLibSymbolRef()->GetExcludedFromSim() );
symbol->SetExcludedFromBOM( symbol->GetLibSymbolRef()->GetExcludedFromBOM() );
symbol->SetExcludedFromBoard( symbol->GetLibSymbolRef()->GetExcludedFromBoard() );
}
if( m_resetPinTextVisibility->GetValue() )
{
symbol->SetShowPinNames( symbol->GetLibSymbolRef()->GetShowPinNames() );
symbol->SetShowPinNumbers( symbol->GetLibSymbolRef()->GetShowPinNumbers() );
}
bool removeExtras = m_removeExtraBox->GetValue();
bool resetVis = m_resetFieldVisibilities->GetValue();
bool resetEffects = m_resetFieldEffects->GetValue();
bool resetPositions = m_resetFieldPositions->GetValue();
for( int ii = (int) symbol->GetFields().size() - 1; ii >= 0; ii-- )
{
SCH_FIELD& field = symbol->GetFields()[ii];
SCH_FIELD* libField = nullptr;
bool doUpdate = field.IsPrivate();
// Mandatory fields always exist in m_updateFields, but these names can be translated.
// so use GetCanonicalName().
doUpdate |= alg::contains( m_updateFields, field.GetCanonicalName() );
if( !doUpdate )
continue;
if( field.IsMandatory() )
libField = symbol->GetLibSymbolRef()->GetField( field.GetId() );
else
libField = symbol->GetLibSymbolRef()->GetField( field.GetName() );
if( libField )
{
field.SetPrivate( libField->IsPrivate() );
bool resetText = libField->GetText().IsEmpty() ? m_resetEmptyFields->GetValue()
: m_resetFieldText->GetValue();
if( resetText )
{
if( field.GetId() == FIELD_T::REFERENCE )
{
wxString prefix = UTIL::GetRefDesPrefix( libField->GetText() );
for( const SCH_SHEET_PATH& instance : symbol_change_info.m_Instances )
{
wxString ref = symbol->GetRef( &instance );
int number = UTIL::GetRefDesNumber( ref );
if( number >= 0 )
ref.Printf( wxS( "%s%d" ), prefix, number );
else
ref = UTIL::GetRefDesUnannotated( prefix );
symbol->SetRef( &instance, ref );
}
}
else if( field.GetId() == FIELD_T::VALUE )
{
if( !symbol->IsPower() || m_resetCustomPower->IsChecked() )
symbol->SetValueFieldText( UnescapeString( libField->GetText() ) );
}
else
{
field.SetText( libField->GetText() );
}
}
if( resetVis )
field.SetVisible( libField->IsVisible() );
if( resetEffects )
{
// Careful: the visible bit and position are also set by SetAttributes()
bool visible = field.IsVisible();
VECTOR2I pos = field.GetPosition();
field.SetAttributes( *libField );
field.SetVisible( visible );
field.SetPosition( pos );
field.SetNameShown( libField->IsNameShown() );
field.SetCanAutoplace( libField->CanAutoplace() );
}
if( resetPositions )
field.SetTextPos( symbol->GetPosition() + libField->GetTextPos() );
}
else if( !field.IsMandatory() && removeExtras )
{
symbol->RemoveField( field.GetName() );
}
}
std::vector<SCH_FIELD*> libFields;
symbol->GetLibSymbolRef()->GetFields( libFields );
for( SCH_FIELD* libField : libFields )
{
if( libField->IsMandatory() )
continue;
if( !alg::contains( m_updateFields, libField->GetCanonicalName() ) )
continue;
if( !symbol->GetField( libField->GetName() ) )
{
SCH_FIELD* schField = symbol->AddField( SCH_FIELD( symbol, FIELD_T::USER, libField->GetName() ) );
// SetAttributes() also covers text angle, size, italic and bold
schField->SetAttributes( *libField );
schField->SetVisible( libField->IsVisible() );
schField->SetText( libField->GetText() );
schField->SetTextPos( symbol->GetPosition() + libField->GetTextPos() );
schField->SetPrivate( libField->IsPrivate() );
}
if( resetPositions && frame->eeconfig()->m_AutoplaceFields.enable )
{
AUTOPLACE_ALGO fieldsAutoplaced = symbol->GetFieldsAutoplaced();
if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
symbol->AutoplaceFields( screen, fieldsAutoplaced );
}
}
symbol->SetSchSymbolLibraryName( wxEmptyString );
screen->Append( symbol );
if( resetPositions )
{
AUTOPLACE_ALGO fieldsAutoplaced = symbol->GetFieldsAutoplaced();
if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
symbol->AutoplaceFields( screen, fieldsAutoplaced );
}
// Clear alternate pins as required.
for( const SCH_SHEET_PATH& instance : symbol_change_info.m_Instances )
{
for( SCH_PIN* pin : symbol->GetPins( &instance ) )
{
if( !pin->GetAlt().IsEmpty() )
{
// There is a bug that allows the alternate pin name to be set to the default pin
// name which causes library symbol comparison errors. Clear the alternate pin
// name in this case even if the reset option is not checked. Also clear the
// alternate pin name if it no longer exists in the alternate pin map.
if( m_resetAlternatePin->IsChecked()
|| ( pin->GetAlt() == pin->GetBaseName() )
|| ( pin->GetAlternates().count( pin->GetAlt() ) == 0 ) )
{
pin->SetAlt( wxEmptyString );
}
}
}
}
frame->GetCanvas()->GetView()->Update( symbol );
msg = getSymbolReferences( *symbol, symbol_change_info.m_LibId, &initialLibLinkName );
msg += wxS( ": OK" );
m_messagePanel->Report( msg, RPT_SEVERITY_ACTION );
matchesProcessed +=1;
}
return matchesProcessed;
}
wxString DIALOG_CHANGE_SYMBOLS::getSymbolReferences( SCH_SYMBOL& aSymbol, const LIB_ID& aNewId,
const wxString* aOldLibLinkName )
{
wxString msg;
wxString references;
const LIB_ID& oldId = aSymbol.GetLibId();
wxString oldLibLinkName; // For report
if( aOldLibLinkName )
oldLibLinkName = *aOldLibLinkName;
else
oldLibLinkName = UnescapeString( oldId.Format() );
SCH_EDIT_FRAME* parent = dynamic_cast< SCH_EDIT_FRAME* >( GetParent() );
wxCHECK( parent, msg );
SCH_SHEET_LIST sheets = parent->Schematic().Hierarchy();
for( const SCH_SYMBOL_INSTANCE& instance : aSymbol.GetInstances() )
{
// Only include the symbol instances for the current project.
if( !sheets.HasPath( instance.m_Path ) )
continue;
if( references.IsEmpty() )
references = instance.m_Reference;
else
references += wxT( " " ) + instance.m_Reference;
}
if( m_mode == MODE::UPDATE )
{
if( aSymbol.GetInstances().size() == 1 )
{
msg.Printf( _( "Update symbol %s from '%s' to '%s'" ),
references,
oldLibLinkName,
UnescapeString( aNewId.Format() ) );
}
else
{
msg.Printf( _( "Update symbols %s from '%s' to '%s'" ),
references,
oldLibLinkName,
UnescapeString( aNewId.Format() ) );
}
}
else // mode is MODE::CHANGE
{
if( aSymbol.GetInstances().size() == 1 )
{
msg.Printf( _( "Change symbol %s from '%s' to '%s'" ),
references,
oldLibLinkName,
UnescapeString( aNewId.Format() ) );
}
else
{
msg.Printf( _( "Change symbols %s from '%s' to '%s'" ),
references,
oldLibLinkName,
UnescapeString( aNewId.Format() ) );
}
}
return msg;
}