Make updating netnames in backannotate more discrete.

Trying to be too smart was producing unexpected results.

Fixes https://gitlab.com/kicad/code/kicad/issues/6480
This commit is contained in:
Jeff Young 2020-11-25 22:16:06 +00:00
parent b8414174af
commit c85ef8eb69
5 changed files with 66 additions and 115 deletions

View File

@ -1359,14 +1359,15 @@ void CONNECTION_GRAPH::buildConnectionGraph()
// Recache remaining valid subgraphs by sheet path // Recache remaining valid subgraphs by sheet path
m_sheet_to_subgraphs_map.clear(); m_sheet_to_subgraphs_map.clear();
for( auto subgraph : m_driver_subgraphs )
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs )
m_sheet_to_subgraphs_map[ subgraph->m_sheet ].emplace_back( subgraph ); m_sheet_to_subgraphs_map[ subgraph->m_sheet ].emplace_back( subgraph );
// Next time through the subgraphs, we do some post-processing to handle things like // Next time through the subgraphs, we do some post-processing to handle things like
// connecting bus members to their neighboring subgraphs, and then propagate connections // connecting bus members to their neighboring subgraphs, and then propagate connections
// through the hierarchy // through the hierarchy
for( auto subgraph : m_driver_subgraphs ) for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs )
{ {
if( !subgraph->m_dirty ) if( !subgraph->m_dirty )
continue; continue;
@ -2038,8 +2039,8 @@ std::vector<const CONNECTION_SUBGRAPH*> CONNECTION_GRAPH::GetBusesNeedingMigrati
} }
CONNECTION_SUBGRAPH* CONNECTION_GRAPH::FindSubgraphByName( CONNECTION_SUBGRAPH* CONNECTION_GRAPH::FindSubgraphByName( const wxString& aNetName,
const wxString& aNetName, const SCH_SHEET_PATH& aPath ) const SCH_SHEET_PATH& aPath )
{ {
auto it = m_net_name_to_subgraphs_map.find( aNetName ); auto it = m_net_name_to_subgraphs_map.find( aNetName );

View File

@ -321,8 +321,7 @@ private:
std::vector<CONNECTION_SUBGRAPH*> m_driver_subgraphs; std::vector<CONNECTION_SUBGRAPH*> m_driver_subgraphs;
// Cache to lookup subgraphs in m_driver_subgraphs by sheet path // Cache to lookup subgraphs in m_driver_subgraphs by sheet path
std::unordered_map<SCH_SHEET_PATH, std::unordered_map<SCH_SHEET_PATH, std::vector<CONNECTION_SUBGRAPH*>> m_sheet_to_subgraphs_map;
std::vector<CONNECTION_SUBGRAPH*>> m_sheet_to_subgraphs_map;
std::vector<std::pair<SCH_SHEET_PATH, SCH_PIN*>> m_invisible_power_pins; std::vector<std::pair<SCH_SHEET_PATH, SCH_PIN*>> m_invisible_power_pins;
@ -337,8 +336,7 @@ private:
std::map< std::pair<SCH_SHEET_PATH, wxString>, std::map< std::pair<SCH_SHEET_PATH, wxString>,
std::vector<const CONNECTION_SUBGRAPH*> > m_local_label_cache; std::vector<const CONNECTION_SUBGRAPH*> > m_local_label_cache;
std::unordered_map<wxString, std::unordered_map<wxString, std::vector<CONNECTION_SUBGRAPH*>> m_net_name_to_subgraphs_map;
std::vector<CONNECTION_SUBGRAPH*>> m_net_name_to_subgraphs_map;
std::map<SCH_ITEM*, CONNECTION_SUBGRAPH*> m_item_to_subgraph_map; std::map<SCH_ITEM*, CONNECTION_SUBGRAPH*> m_item_to_subgraph_map;

View File

@ -310,7 +310,9 @@ bool DIALOG_LABEL_EDITOR::TransferDataFromWindow()
} }
if( !text.IsEmpty() ) if( !text.IsEmpty() )
{
m_CurrentText->SetText( text ); m_CurrentText->SetText( text );
}
else if( !m_CurrentText->IsNew() ) else if( !m_CurrentText->IsNew() )
{ {
DisplayError( this, _( "Empty Text!" ) ); DisplayError( this, _( "Empty Text!" ) );

View File

@ -35,6 +35,7 @@
#include <kicad_string.h> #include <kicad_string.h>
#include <kiface_i.h> #include <kiface_i.h>
#include <wildcards_and_files_ext.h> #include <wildcards_and_files_ext.h>
#include <connection_graph.h>
BACK_ANNOTATE::BACK_ANNOTATE( SCH_EDIT_FRAME* aFrame, REPORTER& aReporter, bool aRelinkFootprints, BACK_ANNOTATE::BACK_ANNOTATE( SCH_EDIT_FRAME* aFrame, REPORTER& aReporter, bool aRelinkFootprints,
bool aProcessFootprints, bool aProcessValues, bool aProcessFootprints, bool aProcessValues,
@ -304,8 +305,7 @@ void BACK_ANNOTATE::checkForUnusedSymbols()
void BACK_ANNOTATE::applyChangelist() void BACK_ANNOTATE::applyChangelist()
{ {
std::set<wxString> handledNetChanges; wxString msg;
wxString msg;
// Apply changes from change list // Apply changes from change list
for( CHANGELIST_ITEM& item : m_changelist ) for( CHANGELIST_ITEM& item : m_changelist )
@ -389,17 +389,13 @@ void BACK_ANNOTATE::applyChangelist()
continue; continue;
} }
SCH_CONNECTION* conn = pin->Connection( &ref.GetSheetPath() ); SCH_CONNECTION* connection = pin->Connection( &ref.GetSheetPath() );
wxString key = shortNetName + ref.GetSheetPath().PathAsString(); if( connection && connection->Name( true ) != shortNetName )
{
if( handledNetChanges.count( key ) ) processNetNameChange( ref.GetFullRef(), pin, connection,
continue; connection->Name( true ), shortNetName );
else }
handledNetChanges.insert( key );
if( conn && conn->Name( true ) != shortNetName )
processNetNameChange( conn, conn->Name( true ), shortNetName );
} }
} }
} }
@ -482,115 +478,65 @@ static LABEL_SPIN_STYLE orientLabel( SCH_PIN* aPin )
} }
void BACK_ANNOTATE::processNetNameChange( SCH_CONNECTION* aConn, const wxString& aOldName, void addConnections( SCH_ITEM* aItem, const SCH_SHEET_PATH& aSheetPath,
const wxString& aNewName ) std::set<SCH_ITEM*>& connectedItems )
{ {
wxString msg; if( connectedItems.insert( aItem ).second )
SCH_ITEM* driver = aConn->Driver(); {
for( SCH_ITEM* connectedItem : aItem->ConnectedItems( aSheetPath ) )
addConnections( connectedItem, aSheetPath, connectedItems );
}
}
auto editMatchingLabels =
[this]( SCH_SCREEN* aScreen, KICAD_T aType, const wxString& oldName,
const wxString& newName )
{
for( SCH_ITEM* schItem : aScreen->Items().OfType( aType ) )
{
SCH_TEXT* label = static_cast<SCH_TEXT*>( schItem );
if( EscapeString( label->GetShownText(), CTX_NETNAME ) == oldName ) void BACK_ANNOTATE::processNetNameChange( const wxString& aRef, SCH_PIN* aPin,
{ const SCH_CONNECTION* aConnection,
m_frame->SaveCopyInUndoList( aScreen, label, UNDO_REDO::CHANGED, const wxString& aOldName, const wxString& aNewName )
m_appendUndo ); {
m_appendUndo = true; wxString msg;
static_cast<SCH_TEXT*>( label )->SetText( newName );
} // Find a physically-connected driver. We can't use the SCH_CONNECTION's m_driver because
} // it has already been resolved by merging subgraphs with the same label, etc., and our
}; // name change may cause that resolution to change.
std::set<SCH_ITEM*> connectedItems;
SCH_ITEM* driver = nullptr;
CONNECTION_SUBGRAPH::PRIORITY driverPriority = CONNECTION_SUBGRAPH::PRIORITY::NONE;
addConnections( aPin, aConnection->Sheet(), connectedItems );
for( SCH_ITEM* item : connectedItems )
{
CONNECTION_SUBGRAPH::PRIORITY priority = CONNECTION_SUBGRAPH::GetDriverPriority( item );
if( priority > driverPriority )
{
driver = item;
driverPriority = priority;
}
}
switch( driver->Type() ) switch( driver->Type() )
{ {
case SCH_LABEL_T: case SCH_LABEL_T:
++m_changesCount;
msg.Printf( _( "Change '%s' labels to '%s'." ), aOldName, aNewName );
if( !m_dryRun )
{
SCH_SHEET_PATH sheet = aConn->Sheet();
SCH_SCREEN* screen = sheet.LastScreen();
for( SCH_ITEM* label : screen->Items().OfType( SCH_LABEL_T ) )
{
SCH_CONNECTION* conn = label->Connection( &sheet );
if( conn && conn->Driver() == driver )
{
m_frame->SaveCopyInUndoList( screen, label, UNDO_REDO::CHANGED, m_appendUndo );
m_appendUndo = true;
static_cast<SCH_TEXT*>( label )->SetText( aNewName );
}
}
}
m_reporter.ReportHead( msg, RPT_SEVERITY_ACTION );
break;
case SCH_GLOBAL_LABEL_T: case SCH_GLOBAL_LABEL_T:
++m_changesCount;
msg.Printf( _( "Change '%s' global labels to '%s'." ), aOldName, aNewName );
if( !m_dryRun )
{
for( const SCH_SHEET_PATH& sheet : m_frame->Schematic().GetSheets() )
editMatchingLabels( sheet.LastScreen(), SCH_GLOBAL_LABEL_T, aOldName, aNewName );
}
m_reporter.ReportHead( msg, RPT_SEVERITY_ACTION );
break;
case SCH_HIER_LABEL_T: case SCH_HIER_LABEL_T:
++m_changesCount;
msg.Printf( _( "Change '%s' hierarchical label to '%s'." ), aOldName, aNewName );
if( !m_dryRun )
{
SCH_SCREEN* screen = aConn->Sheet().LastScreen();
editMatchingLabels( screen, SCH_HIER_LABEL_T, aOldName, aNewName );
SCH_SHEET* sheet = dynamic_cast<SCH_SHEET*>( driver->GetParent() );
wxASSERT( sheet );
if( !sheet )
break;
screen = sheet->GetScreen();
for( SCH_SHEET_PIN* pin : sheet->GetPins() )
{
if( EscapeString( pin->GetShownText(), CTX_NETNAME ) == aOldName )
{
m_frame->SaveCopyInUndoList( screen, pin, UNDO_REDO::CHANGED, m_appendUndo );
m_appendUndo = true;
static_cast<SCH_TEXT*>( pin )->SetText( aNewName );
}
}
}
m_reporter.ReportHead( msg, RPT_SEVERITY_ACTION );
break;
case SCH_SHEET_PIN_T: case SCH_SHEET_PIN_T:
++m_changesCount; ++m_changesCount;
msg.Printf( _( "Change '%s' hierarchical label to '%s'." ), aOldName, aNewName );
msg.Printf( _( "Change %s pin %s net label from '%s' to '%s'." ),
aRef,
aPin->GetNumber(),
aOldName,
aNewName );
if( !m_dryRun ) if( !m_dryRun )
{ {
SCH_SCREEN* screen = aConn->Sheet().LastScreen(); SCH_SCREEN* screen = aConnection->Sheet().LastScreen();
m_frame->SaveCopyInUndoList( screen, driver, UNDO_REDO::CHANGED, m_appendUndo ); m_frame->SaveCopyInUndoList( screen, driver, UNDO_REDO::CHANGED, m_appendUndo );
m_appendUndo = true; m_appendUndo = true;
static_cast<SCH_TEXT*>( driver )->SetText( aNewName ); static_cast<SCH_TEXT*>( driver )->SetText( aNewName );
SCH_SHEET* sheet = static_cast<SCH_SHEET_PIN*>( driver )->GetParent();
screen = sheet->GetScreen();
editMatchingLabels( screen, SCH_HIER_LABEL_T, aOldName, aNewName );
} }
m_reporter.ReportHead( msg, RPT_SEVERITY_ACTION ); m_reporter.ReportHead( msg, RPT_SEVERITY_ACTION );
@ -612,7 +558,10 @@ void BACK_ANNOTATE::processNetNameChange( SCH_CONNECTION* aConn, const wxString&
} }
++m_changesCount; ++m_changesCount;
msg.Printf( _( "Add label '%s' to net %s." ), aNewName, aOldName ); msg.Printf( _( "Add label '%s' to %s pin %s net." ),
aNewName,
aRef,
aPin->GetNumber() );
if( !m_dryRun ) if( !m_dryRun )
{ {
@ -624,7 +573,7 @@ void BACK_ANNOTATE::processNetNameChange( SCH_CONNECTION* aConn, const wxString&
label->SetLabelSpinStyle( spin ); label->SetLabelSpinStyle( spin );
label->SetFlags( IS_NEW ); label->SetFlags( IS_NEW );
SCH_SCREEN* screen = aConn->Sheet().LastScreen(); SCH_SCREEN* screen = aConnection->Sheet().LastScreen();
m_frame->AddItemToScreenAndUndoList( screen, label, m_appendUndo ); m_frame->AddItemToScreenAndUndoList( screen, label, m_appendUndo );
m_appendUndo = true; m_appendUndo = true;
} }

View File

@ -141,7 +141,8 @@ private:
*/ */
void applyChangelist(); void applyChangelist();
void processNetNameChange( SCH_CONNECTION* aConn, const wxString& aOldName, void processNetNameChange( const wxString& aRef, SCH_PIN* aPin,
const SCH_CONNECTION* aConnection, const wxString& aOldName,
const wxString& aNewName ); const wxString& aNewName );
}; };