/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright The KiCad Developers, see AUTHORS.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 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 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include bool SCHEMATIC::m_IsSchematicExists = false; SCHEMATIC::SCHEMATIC( PROJECT* aPrj ) : EDA_ITEM( nullptr, SCHEMATIC_T ), m_project( nullptr ), m_rootSheet( nullptr ), m_schematicHolder( nullptr ) { m_currentSheet = new SCH_SHEET_PATH(); m_connectionGraph = new CONNECTION_GRAPH( this ); m_IsSchematicExists = true; SetProject( aPrj ); PROPERTY_MANAGER::Instance().RegisterListener( TYPE_HASH( SCH_FIELD ), [&]( INSPECTABLE* aItem, PROPERTY_BASE* aProperty, COMMIT* aCommit ) { // Special case: propagate value, footprint, and datasheet fields to other units // of a given symbol if they aren't in the selection SCH_FIELD* field = dynamic_cast( aItem ); if( !field || !IsValid() ) return; SCH_SYMBOL* symbol = dynamic_cast( field->GetParent() ); if( !symbol || aProperty->Name() != _HKI( "Text" ) ) return; // TODO(JE) This will need to get smarter to enable API access SCH_SHEET_PATH sheetPath = CurrentSheet(); wxString newValue = aItem->Get( aProperty ); if( field->GetId() == FIELD_T::REFERENCE ) { symbol->SetRef( &sheetPath, newValue ); // The user might want to change all the units to the new ref. Or they // might not. Since we have no way of knowing, we default to the most // concrete action (change only the selected reference). return; } wxString ref = symbol->GetRef( &sheetPath ); int unit = symbol->GetUnit(); LIB_ID libId = symbol->GetLibId(); for( SCH_SHEET_PATH& sheet : Hierarchy() ) { std::vector otherUnits; CollectOtherUnits( ref, unit, libId, sheet, &otherUnits ); for( SCH_SYMBOL* otherUnit : otherUnits ) { switch( field->GetId() ) { case FIELD_T::VALUE: case FIELD_T::FOOTPRINT: case FIELD_T::DATASHEET: { if( aCommit ) aCommit->Modify( otherUnit, sheet.LastScreen() ); otherUnit->GetField( field->GetId() )->SetText( newValue ); break; } default: break; } } } } ); } SCHEMATIC::~SCHEMATIC() { PROPERTY_MANAGER::Instance().UnregisterListeners( TYPE_HASH( SCH_FIELD ) ); delete m_currentSheet; delete m_connectionGraph; m_IsSchematicExists = false; } void SCHEMATIC::Reset() { delete m_rootSheet; m_rootSheet = nullptr; m_connectionGraph->Reset(); m_currentSheet->clear(); } void SCHEMATIC::SetProject( PROJECT* aPrj ) { if( m_project ) { PROJECT_FILE& project = m_project->GetProjectFile(); // d'tor will save settings to file delete project.m_ErcSettings; project.m_ErcSettings = nullptr; // d'tor will save settings to file delete project.m_SchematicSettings; project.m_SchematicSettings = nullptr; } m_project = aPrj; if( m_project ) { PROJECT_FILE& project = m_project->GetProjectFile(); project.m_ErcSettings = new ERC_SETTINGS( &project, "erc" ); project.m_SchematicSettings = new SCHEMATIC_SETTINGS( &project, "schematic" ); project.m_SchematicSettings->LoadFromFile(); project.m_SchematicSettings->m_NgspiceSettings->LoadFromFile(); project.m_ErcSettings->LoadFromFile(); } } void SCHEMATIC::CacheExistingAnnotation() { wxASSERT( m_project ); // Cache all existing annotations in the REFDES_TRACKER std::shared_ptr refdesTracker = m_project->GetProjectFile().m_SchematicSettings->m_refDesTracker; SCH_SHEET_LIST sheets = Hierarchy(); SCH_REFERENCE_LIST references; sheets.GetSymbols( references ); for( const SCH_REFERENCE& ref : references ) { refdesTracker->Insert( ref.GetFullRef( false ).ToStdString() ); } } bool SCHEMATIC::Contains( const SCH_REFERENCE& aRef ) const { SCH_SHEET_LIST sheets = Hierarchy(); SCH_REFERENCE_LIST references; /// TODO(snh): This is horribly inefficient, we should be using refdesTracker for this. /// REFDES_TRACKER will need to be extended to track if a reference is currently present in the schematic /// as well as the units. For now, this is relatively fast for reasonably sized schematics /// Famous last words... sheets.GetSymbols( references ); return std::any_of( references.begin(), references.end(), [&]( const SCH_REFERENCE& ref ) { return ref.GetFullRef( true ) == aRef.GetFullRef( true ); } ); } void SCHEMATIC::SetRoot( SCH_SHEET* aRootSheet ) { wxCHECK_RET( aRootSheet, wxS( "Call to SetRoot with null SCH_SHEET!" ) ); m_rootSheet = aRootSheet; m_currentSheet->clear(); m_currentSheet->push_back( m_rootSheet ); m_hierarchy = BuildSheetListSortedByPageNumbers(); m_connectionGraph->Reset(); } SCH_SCREEN* SCHEMATIC::RootScreen() const { return IsValid() ? m_rootSheet->GetScreen() : nullptr; } SCH_SHEET_LIST SCHEMATIC::Hierarchy() const { wxCHECK( !m_hierarchy.empty(), m_hierarchy ); return m_hierarchy; } void SCHEMATIC::RefreshHierarchy() { m_hierarchy = BuildSheetListSortedByPageNumbers(); } void SCHEMATIC::GetContextualTextVars( wxArrayString* aVars ) const { auto add = [&]( const wxString& aVar ) { if( !alg::contains( *aVars, aVar ) ) aVars->push_back( aVar ); }; add( wxT( "#" ) ); add( wxT( "##" ) ); add( wxT( "SHEETPATH" ) ); add( wxT( "SHEETNAME" ) ); add( wxT( "FILENAME" ) ); add( wxT( "FILEPATH" ) ); add( wxT( "PROJECTNAME" ) ); if( !CurrentSheet().empty() ) CurrentSheet().LastScreen()->GetTitleBlock().GetContextualTextVars( aVars ); for( std::pair entry : m_project->GetTextVars() ) add( entry.first ); } bool SCHEMATIC::ResolveTextVar( const SCH_SHEET_PATH* aSheetPath, wxString* token, int aDepth ) const { wxCHECK( aSheetPath, false ); if( token->IsSameAs( wxT( "#" ) ) ) { *token = aSheetPath->GetPageNumber(); return true; } else if( token->IsSameAs( wxT( "##" ) ) ) { *token = wxString::Format( "%i", Root().CountSheets() ); return true; } else if( token->IsSameAs( wxT( "SHEETPATH" ) ) ) { *token = aSheetPath->PathHumanReadable(); return true; } else if( token->IsSameAs( wxT( "SHEETNAME" ) ) ) { *token = aSheetPath->Last()->GetName(); return true; } else if( token->IsSameAs( wxT( "FILENAME" ) ) ) { wxFileName fn( GetFileName() ); *token = fn.GetFullName(); return true; } else if( token->IsSameAs( wxT( "FILEPATH" ) ) ) { wxFileName fn( GetFileName() ); *token = fn.GetFullPath(); return true; } else if( token->IsSameAs( wxT( "PROJECTNAME" ) ) ) { *token = m_project->GetProjectName(); return true; } if( aSheetPath->LastScreen()->GetTitleBlock().TextVarResolver( token, m_project ) ) return true; if( m_project->TextVarResolver( token ) ) return true; return false; } wxString SCHEMATIC::GetFileName() const { return IsValid() ? m_rootSheet->GetScreen()->GetFileName() : wxString( wxEmptyString ); } SCHEMATIC_SETTINGS& SCHEMATIC::Settings() const { wxASSERT( m_project ); return *m_project->GetProjectFile().m_SchematicSettings; } ERC_SETTINGS& SCHEMATIC::ErcSettings() const { wxASSERT( m_project ); return *m_project->GetProjectFile().m_ErcSettings; } std::vector SCHEMATIC::ResolveERCExclusions() { SCH_SHEET_LIST sheetList = Hierarchy(); ERC_SETTINGS& settings = ErcSettings(); // Migrate legacy marker exclusions to new format to ensure exclusion matching functions across // file versions. Silently drops any legacy exclusions which can not be mapped to the new format // without risking an incorrect exclusion - this is preferable to silently dropping // new ERC errors / warnings due to an incorrect match between a legacy and new // marker serialization format std::set migratedExclusions; for( auto it = settings.m_ErcExclusions.begin(); it != settings.m_ErcExclusions.end(); ) { SCH_MARKER* testMarker = SCH_MARKER::DeserializeFromString( sheetList, *it ); if( !testMarker ) { it = settings.m_ErcExclusions.erase( it ); continue; } if( testMarker->IsLegacyMarker() ) { const wxString settingsKey = testMarker->GetRCItem()->GetSettingsKey(); if( settingsKey != wxT( "pin_to_pin" ) && settingsKey != wxT( "hier_label_mismatch" ) && settingsKey != wxT( "different_unit_net" ) ) { migratedExclusions.insert( testMarker->SerializeToString() ); } it = settings.m_ErcExclusions.erase( it ); } else { ++it; } delete testMarker; } settings.m_ErcExclusions.insert( migratedExclusions.begin(), migratedExclusions.end() ); // End of legacy exclusion removal / migrations for( const SCH_SHEET_PATH& sheet : sheetList ) { for( SCH_ITEM* item : sheet.LastScreen()->Items().OfType( SCH_MARKER_T ) ) { SCH_MARKER* marker = static_cast( item ); wxString serialized = marker->SerializeToString(); std::set::iterator it = settings.m_ErcExclusions.find( serialized ); if( it != settings.m_ErcExclusions.end() ) { marker->SetExcluded( true, settings.m_ErcExclusionComments[serialized] ); settings.m_ErcExclusions.erase( it ); } } } std::vector newMarkers; for( const wxString& serialized : settings.m_ErcExclusions ) { SCH_MARKER* marker = SCH_MARKER::DeserializeFromString( sheetList, serialized ); if( marker ) { marker->SetExcluded( true, settings.m_ErcExclusionComments[serialized] ); newMarkers.push_back( marker ); } } settings.m_ErcExclusions.clear(); return newMarkers; } std::shared_ptr SCHEMATIC::GetBusAlias( const wxString& aLabel ) const { for( const SCH_SHEET_PATH& sheet : Hierarchy() ) { for( const std::shared_ptr& alias : sheet.LastScreen()->GetBusAliases() ) { if( alias->GetName() == aLabel ) return alias; } } return nullptr; } std::set SCHEMATIC::GetNetClassAssignmentCandidates() { std::set names; for( const auto& [ key, subgraphList ] : m_connectionGraph->GetNetMap() ) { CONNECTION_SUBGRAPH* firstSubgraph = subgraphList[0]; if( !firstSubgraph->GetDriverConnection()->IsBus() && firstSubgraph->GetDriverPriority() >= CONNECTION_SUBGRAPH::PRIORITY::PIN ) { names.insert( key.Name ); } } return names; } bool SCHEMATIC::ResolveCrossReference( wxString* token, int aDepth ) const { wxString remainder; wxString ref = token->BeforeFirst( ':', &remainder ); KIID_PATH path( ref ); KIID uuid = path.back(); SCH_SHEET_PATH sheetPath; SCH_ITEM* refItem = ResolveItem( KIID( uuid ), &sheetPath, true ); if( path.size() > 1 ) { path.pop_back(); sheetPath = Hierarchy().GetSheetPathByKIIDPath( path ).value_or( sheetPath ); } if( refItem && refItem->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* refSymbol = static_cast( refItem ); if( refSymbol->ResolveTextVar( &sheetPath, &remainder, aDepth + 1 ) ) *token = std::move( remainder ); else *token = refSymbol->GetRef( &sheetPath, true ) + wxS( ":" ) + remainder; return true; // Cross-reference is resolved whether or not the actual textvar was } else if( refItem && refItem->Type() == SCH_SHEET_T ) { SCH_SHEET* refSheet = static_cast( refItem ); sheetPath.push_back( refSheet ); if( refSheet->ResolveTextVar( &sheetPath, &remainder, aDepth + 1 ) ) *token = std::move( remainder ); return true; // Cross-reference is resolved whether or not the actual textvar was } return false; } std::map SCHEMATIC::GetVirtualPageToSheetNamesMap() const { std::map namesMap; for( const SCH_SHEET_PATH& sheet : Hierarchy() ) { if( sheet.size() == 1 ) namesMap[sheet.GetVirtualPageNumber()] = _( "" ); else namesMap[sheet.GetVirtualPageNumber()] = sheet.Last()->GetName(); } return namesMap; } std::map SCHEMATIC::GetVirtualPageToSheetPagesMap() const { std::map pagesMap; for( const SCH_SHEET_PATH& sheet : Hierarchy() ) pagesMap[sheet.GetVirtualPageNumber()] = sheet.GetPageNumber(); return pagesMap; } wxString SCHEMATIC::ConvertRefsToKIIDs( const wxString& aSource ) const { wxString newbuf; size_t sourceLen = aSource.length(); for( size_t i = 0; i < sourceLen; ++i ) { if( aSource[i] == '$' && i + 1 < sourceLen && aSource[i+1] == '{' ) { wxString token; bool isCrossRef = false; int nesting = 0; for( i = i + 2; i < sourceLen; ++i ) { if( aSource[i] == '{' && ( aSource[i-1] == '_' || aSource[i-1] == '^' || aSource[i-1] == '~' ) ) { nesting++; } if( aSource[i] == '}' ) { nesting--; if( nesting < 0 ) break; } if( aSource[i] == ':' ) isCrossRef = true; token.append( aSource[i] ); } if( isCrossRef ) { wxString remainder; wxString ref = token.BeforeFirst( ':', &remainder ); SCH_REFERENCE_LIST references; Hierarchy().GetSymbols( references ); for( size_t jj = 0; jj < references.GetCount(); jj++ ) { SCH_SYMBOL* refSymbol = references[ jj ].GetSymbol(); if( ref == refSymbol->GetRef( &references[ jj ].GetSheetPath(), true ) ) { KIID_PATH path = references[ jj ].GetSheetPath().Path(); path.push_back( refSymbol->m_Uuid ); token = path.AsString() + wxS( ":" ) + remainder; break; } } } newbuf.append( wxS( "${" ) + token + wxS( "}" ) ); } else { newbuf.append( aSource[i] ); } } return newbuf; } wxString SCHEMATIC::ConvertKIIDsToRefs( const wxString& aSource ) const { wxString newbuf; size_t sourceLen = aSource.length(); for( size_t i = 0; i < sourceLen; ++i ) { if( aSource[i] == '$' && i + 1 < sourceLen && aSource[i+1] == '{' ) { wxString token; bool isCrossRef = false; for( i = i + 2; i < sourceLen; ++i ) { if( aSource[i] == '}' ) break; if( aSource[i] == ':' ) isCrossRef = true; token.append( aSource[i] ); } if( isCrossRef ) { wxString remainder; wxString ref = token.BeforeFirst( ':', &remainder ); KIID_PATH path( ref ); KIID uuid = path.back(); SCH_SHEET_PATH sheetPath; SCH_ITEM* refItem = ResolveItem( uuid, &sheetPath, true ); if( path.size() > 1 ) { path.pop_back(); sheetPath = Hierarchy().GetSheetPathByKIIDPath( path ).value_or( sheetPath ); } if( refItem && refItem->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* refSymbol = static_cast( refItem ); token = refSymbol->GetRef( &sheetPath, true ) + wxS( ":" ) + remainder; } } newbuf.append( wxS( "${" ) + token + wxS( "}" ) ); } else { newbuf.append( aSource[i] ); } } return newbuf; } void SCHEMATIC::SetLegacySymbolInstanceData() { SCH_SCREENS screens( m_rootSheet ); screens.SetLegacySymbolInstanceData(); } wxString SCHEMATIC::GetUniqueFilenameForCurrentSheet() { // Filename is rootSheetName-sheetName-...-sheetName // Note that we need to fetch the rootSheetName out of its filename, as the root SCH_SHEET's // name is just a timestamp. wxFileName rootFn( CurrentSheet().at( 0 )->GetFileName() ); wxString filename = rootFn.GetName(); for( unsigned i = 1; i < CurrentSheet().size(); i++ ) filename += wxT( "-" ) + CurrentSheet().at( i )->GetName(); return filename; } void SCHEMATIC::SetSheetNumberAndCount() { SCH_SCREEN* screen; SCH_SCREENS s_list( Root() ); // Set the sheet count, and the sheet number (1 for root sheet) int sheet_count = Root().CountSheets(); int sheet_number = 1; const KIID_PATH& current_sheetpath = CurrentSheet().Path(); // @todo Remove all pseudo page number system is left over from prior to real page number // implementation. for( const SCH_SHEET_PATH& sheet : Hierarchy() ) { if( sheet.Path() == current_sheetpath ) // Current sheet path found break; sheet_number++; // Not found, increment before this current path } for( screen = s_list.GetFirst(); screen != nullptr; screen = s_list.GetNext() ) screen->SetPageCount( sheet_count ); CurrentSheet().SetVirtualPageNumber( sheet_number ); CurrentSheet().LastScreen()->SetVirtualPageNumber( sheet_number ); CurrentSheet().LastScreen()->SetPageNumber( CurrentSheet().GetPageNumber() ); } void SCHEMATIC::RecomputeIntersheetRefs() { std::map>& pageRefsMap = GetPageRefsMap(); pageRefsMap.clear(); for( const SCH_SHEET_PATH& sheet : Hierarchy() ) { for( SCH_ITEM* item : sheet.LastScreen()->Items().OfType( SCH_GLOBAL_LABEL_T ) ) { SCH_GLOBALLABEL* global = static_cast( item ); wxString resolvedLabel = global->GetShownText( &sheet, false ); pageRefsMap[ resolvedLabel ].insert( sheet.GetVirtualPageNumber() ); } } bool show = Settings().m_IntersheetRefsShow; // Refresh all visible global labels. Note that we have to collect them first as the // SCH_SCREEN::Update() call is going to invalidate the RTree iterator. std::vector currentSheetGlobalLabels; for( EDA_ITEM* item : CurrentSheet().LastScreen()->Items().OfType( SCH_GLOBAL_LABEL_T ) ) currentSheetGlobalLabels.push_back( static_cast( item ) ); for( SCH_GLOBALLABEL* globalLabel : currentSheetGlobalLabels ) { std::vector& fields = globalLabel->GetFields(); fields[0].SetVisible( show ); if( show ) { if( fields.size() == 1 && fields[0].GetTextPos() == globalLabel->GetPosition() ) globalLabel->AutoplaceFields( CurrentSheet().LastScreen(), AUTOPLACE_AUTO ); CurrentSheet().LastScreen()->Update( globalLabel ); for( SCH_FIELD& field : globalLabel->GetFields() ) field.ClearBoundingBoxCache(); globalLabel->ClearBoundingBoxCache(); if( m_schematicHolder ) m_schematicHolder->IntersheetRefUpdate( globalLabel ); } } } wxString SCHEMATIC::GetOperatingPoint( const wxString& aNetName, int aPrecision, const wxString& aRange ) { wxString spiceNetName( aNetName.Lower() ); NETLIST_EXPORTER_SPICE::ConvertToSpiceMarkup( &spiceNetName ); if( spiceNetName == wxS( "gnd" ) || spiceNetName == wxS( "0" ) ) return wxEmptyString; auto it = m_operatingPoints.find( spiceNetName ); if( it != m_operatingPoints.end() ) return SPICE_VALUE( it->second ).ToString( { aPrecision, aRange } ); else if( m_operatingPoints.empty() ) return wxS( "--" ); else return wxS( "?" ); } void SCHEMATIC::FixupJunctionsAfterImport() { SCH_SCREENS screens( Root() ); for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() ) { std::deque allItems; for( SCH_ITEM* item : screen->Items() ) allItems.push_back( item ); // Add missing junctions and breakup wires as needed for( const VECTOR2I& point : screen->GetNeededJunctions( allItems ) ) { SCH_JUNCTION* junction = new SCH_JUNCTION( point ); screen->Append( junction ); // Breakup wires for( SCH_LINE* wire : screen->GetBusesAndWires( point, true ) ) { SCH_LINE* newSegment = wire->NonGroupAware_BreakAt( point ); screen->Append( newSegment ); } } } } void SCHEMATIC::OnItemsAdded( std::vector& aNewItems ) { InvokeListeners( &SCHEMATIC_LISTENER::OnSchItemsAdded, *this, aNewItems ); } void SCHEMATIC::OnItemsRemoved( std::vector& aRemovedItems ) { InvokeListeners( &SCHEMATIC_LISTENER::OnSchItemsRemoved, *this, aRemovedItems ); } void SCHEMATIC::OnItemsChanged( std::vector& aItems ) { InvokeListeners( &SCHEMATIC_LISTENER::OnSchItemsChanged, *this, aItems ); } void SCHEMATIC::OnSchSheetChanged() { InvokeListeners( &SCHEMATIC_LISTENER::OnSchSheetChanged, *this ); } void SCHEMATIC::AddListener( SCHEMATIC_LISTENER* aListener ) { if( !alg::contains( m_listeners, aListener ) ) m_listeners.push_back( aListener ); } void SCHEMATIC::RemoveListener( SCHEMATIC_LISTENER* aListener ) { auto i = std::find( m_listeners.begin(), m_listeners.end(), aListener ); if( i != m_listeners.end() ) { std::iter_swap( i, m_listeners.end() - 1 ); m_listeners.pop_back(); } } void SCHEMATIC::RemoveAllListeners() { m_listeners.clear(); } void SCHEMATIC::RecordERCExclusions() { // Use a sorted sheetList to reduce file churn SCH_SHEET_LIST sheetList = Hierarchy(); ERC_SETTINGS& ercSettings = ErcSettings(); ercSettings.m_ErcExclusions.clear(); ercSettings.m_ErcExclusionComments.clear(); for( unsigned i = 0; i < sheetList.size(); i++ ) { for( SCH_ITEM* item : sheetList[i].LastScreen()->Items().OfType( SCH_MARKER_T ) ) { SCH_MARKER* marker = static_cast( item ); if( marker->IsExcluded() ) { wxString serialized = marker->SerializeToString(); ercSettings.m_ErcExclusions.insert( serialized ); ercSettings.m_ErcExclusionComments[ serialized ] = marker->GetComment(); } } } } void SCHEMATIC::ResolveERCExclusionsPostUpdate() { SCH_SHEET_LIST sheetList = Hierarchy(); for( SCH_MARKER* marker : ResolveERCExclusions() ) { SCH_SHEET_PATH errorPath; ignore_unused( sheetList.ResolveItem( marker->GetRCItem()->GetMainItemID(), &errorPath ) ); if( errorPath.LastScreen() ) errorPath.LastScreen()->Append( marker ); else RootScreen()->Append( marker ); } // Once we have the ERC Exclusions, record them in the project file so that // they are retained even before the schematic is saved (PCB Editor can also save the project) RecordERCExclusions(); } EMBEDDED_FILES* SCHEMATIC::GetEmbeddedFiles() { return static_cast( this ); } const EMBEDDED_FILES* SCHEMATIC::GetEmbeddedFiles() const { return static_cast( this ); } void SCHEMATIC::RunOnNestedEmbeddedFiles( const std::function& aFunction ) { SCH_SCREENS screens( Root() ); for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() ) { for( auto& [name, libSym] : screen->GetLibSymbols() ) aFunction( libSym->GetEmbeddedFiles() ); } } std::set SCHEMATIC::GetFonts() const { std::set fonts; SCH_SHEET_LIST sheetList = Hierarchy(); for( const SCH_SHEET_PATH& sheet : sheetList ) { for( SCH_ITEM* item : sheet.LastScreen()->Items() ) { if( EDA_TEXT* text = dynamic_cast( item ) ) { KIFONT::FONT* font = text->GetFont(); if( !font || font->IsStroke() ) continue; using EMBEDDING_PERMISSION = KIFONT::OUTLINE_FONT::EMBEDDING_PERMISSION; auto* outline = static_cast( font ); if( outline->GetEmbeddingPermission() == EMBEDDING_PERMISSION::EDITABLE || outline->GetEmbeddingPermission() == EMBEDDING_PERMISSION::INSTALLABLE ) { fonts.insert( outline ); } } } } return fonts; } void SCHEMATIC::EmbedFonts() { std::set fonts = GetFonts(); for( KIFONT::OUTLINE_FONT* font : fonts ) { auto file = GetEmbeddedFiles()->AddFile( font->GetFileName(), false ); if( !file ) { wxLogTrace( "EMBED", "Failed to add font file: %s", font->GetFileName() ); continue; } file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::FONT; } } std::set SCHEMATIC::GetSchematicsSharedByMultipleProjects() const { std::set retv; wxCHECK( m_rootSheet, retv ); SCH_SHEET_LIST hierarchy( m_rootSheet ); SCH_SCREENS screens( m_rootSheet ); for( const SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() ) { for( const SCH_ITEM* item : screen->Items().OfType( SCH_SYMBOL_T ) ) { const SCH_SYMBOL* symbol = static_cast( item ); const std::vector symbolInstances = symbol->GetInstances(); for( const SCH_SYMBOL_INSTANCE& instance : symbolInstances ) { if( !hierarchy.HasPath( instance.m_Path ) ) { retv.insert( screen ); break; } } if( retv.count( screen ) ) break; } } return retv; } bool SCHEMATIC::IsComplexHierarchy() const { wxCHECK( m_rootSheet, false ); SCH_SCREENS screens( m_rootSheet ); for( const SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() ) { wxCHECK2( screen, continue ); if( screen->GetRefCount() > 1 ) return true; } return false; } void SCHEMATIC::CleanUp( SCH_COMMIT* aCommit, SCH_SCREEN* aScreen ) { SCH_SELECTION_TOOL* selectionTool = m_schematicHolder ? m_schematicHolder->GetSelectionTool() : nullptr; std::vector lines; std::vector junctions; std::vector ncs; std::vector items_to_remove; bool changed = true; if( aScreen == nullptr ) aScreen = GetCurrentScreen(); auto remove_item = [&]( SCH_ITEM* aItem ) -> void { changed = true; if( !( aItem->GetFlags() & STRUCT_DELETED ) ) { aItem->SetFlags( STRUCT_DELETED ); if( aItem->IsSelected() && selectionTool ) selectionTool->RemoveItemFromSel( aItem, true /*quiet mode*/ ); if( m_schematicHolder ) { m_schematicHolder->RemoveFromScreen( aItem, aScreen ); } aCommit->Removed( aItem, aScreen ); } }; for( SCH_ITEM* item : aScreen->Items().OfType( SCH_JUNCTION_T ) ) { if( !aScreen->IsExplicitJunction( item->GetPosition() ) ) items_to_remove.push_back( item ); else junctions.push_back( static_cast( item ) ); } for( SCH_ITEM* item : items_to_remove ) remove_item( item ); for( SCH_ITEM* item : aScreen->Items().OfType( SCH_NO_CONNECT_T ) ) ncs.push_back( static_cast( item ) ); alg::for_all_pairs( junctions.begin(), junctions.end(), [&]( SCH_JUNCTION* aFirst, SCH_JUNCTION* aSecond ) { if( ( aFirst->GetEditFlags() & STRUCT_DELETED ) || ( aSecond->GetEditFlags() & STRUCT_DELETED ) ) { return; } if( aFirst->GetPosition() == aSecond->GetPosition() ) remove_item( aSecond ); } ); alg::for_all_pairs( ncs.begin(), ncs.end(), [&]( SCH_NO_CONNECT* aFirst, SCH_NO_CONNECT* aSecond ) { if( ( aFirst->GetEditFlags() & STRUCT_DELETED ) || ( aSecond->GetEditFlags() & STRUCT_DELETED ) ) { return; } if( aFirst->GetPosition() == aSecond->GetPosition() ) remove_item( aSecond ); } ); auto minX = []( const SCH_LINE* l ) { return std::min( l->GetStartPoint().x, l->GetEndPoint().x ); }; auto maxX = []( const SCH_LINE* l ) { return std::max( l->GetStartPoint().x, l->GetEndPoint().x ); }; auto minY = []( const SCH_LINE* l ) { return std::min( l->GetStartPoint().y, l->GetEndPoint().y ); }; auto maxY = []( const SCH_LINE* l ) { return std::max( l->GetStartPoint().y, l->GetEndPoint().y ); }; // Would be nice to put lines in a canonical form here by swapping // start <-> end as needed but I don't know what swapping breaks. while( changed ) { changed = false; lines.clear(); for( SCH_ITEM* item : aScreen->Items().OfType( SCH_LINE_T ) ) { if( item->GetLayer() == LAYER_WIRE || item->GetLayer() == LAYER_BUS ) lines.push_back( static_cast( item ) ); } // Sort by minimum X position std::sort( lines.begin(), lines.end(), [&]( const SCH_LINE* a, const SCH_LINE* b ) { return minX( a ) < minX( b ); } ); for( auto it1 = lines.begin(); it1 != lines.end(); ++it1 ) { SCH_LINE* firstLine = *it1; if( firstLine->GetEditFlags() & STRUCT_DELETED ) continue; if( firstLine->IsNull() ) { remove_item( firstLine ); continue; } int firstRightXEdge = maxX( firstLine ); auto it2 = it1; for( ++it2; it2 != lines.end(); ++it2 ) { SCH_LINE* secondLine = *it2; int secondLeftXEdge = minX( secondLine ); // impossible to overlap remaining lines if( secondLeftXEdge > firstRightXEdge ) break; // No Y axis overlap if( !( std::max( minY( firstLine ), minY( secondLine ) ) <= std::min( maxY( firstLine ), maxY( secondLine ) ) ) ) { continue; } if( secondLine->GetFlags() & STRUCT_DELETED ) continue; if( !secondLine->IsParallel( firstLine ) || !secondLine->IsStrokeEquivalent( firstLine ) || secondLine->GetLayer() != firstLine->GetLayer() ) { continue; } // Remove identical lines if( firstLine->IsEndPoint( secondLine->GetStartPoint() ) && firstLine->IsEndPoint( secondLine->GetEndPoint() ) ) { remove_item( secondLine ); continue; } // See if we can merge an overlap (or two colinear touching segments with // no junction where they meet). SCH_LINE* mergedLine = secondLine->MergeOverlap( aScreen, firstLine, true ); if( mergedLine != nullptr ) { remove_item( firstLine ); remove_item( secondLine ); if( m_schematicHolder ) { m_schematicHolder->AddToScreen( mergedLine, aScreen ); } aCommit->Added( mergedLine, aScreen ); if( selectionTool && ( firstLine->IsSelected() || secondLine->IsSelected() ) ) selectionTool->AddItemToSel( mergedLine, true /*quiet mode*/ ); break; } } } } } void SCHEMATIC::RecalculateConnections( SCH_COMMIT* aCommit, SCH_CLEANUP_FLAGS aCleanupFlags, TOOL_MANAGER* aToolManager, PROGRESS_REPORTER* aProgressReporter, KIGFX::SCH_VIEW* aSchView, std::function* aChangedItemHandler, PICKED_ITEMS_LIST* aLastChangeList ) { SCHEMATIC_SETTINGS& settings = Settings(); RefreshHierarchy(); SCH_SHEET_LIST list = Hierarchy(); SCH_COMMIT localCommit( aToolManager ); if( !aCommit ) aCommit = &localCommit; PROF_TIMER timer; // Ensure schematic graph is accurate if( aCleanupFlags == LOCAL_CLEANUP ) { CleanUp( aCommit, GetCurrentScreen() ); } else if( aCleanupFlags == GLOBAL_CLEANUP ) { for( const SCH_SHEET_PATH& sheet : list ) CleanUp( aCommit, sheet.LastScreen() ); } timer.Stop(); wxLogTrace( "CONN_PROFILE", "SchematicCleanUp() %0.4f ms", timer.msecs() ); if( settings.m_IntersheetRefsShow ) RecomputeIntersheetRefs(); if( !ADVANCED_CFG::GetCfg().m_IncrementalConnectivity || aCleanupFlags == GLOBAL_CLEANUP || aLastChangeList == nullptr || ConnectionGraph()->IsMinor() ) { // Clear all resolved netclass caches in case labels have changed m_project->GetProjectFile().NetSettings()->ClearAllCaches(); // Update all rule areas so we can cascade implied connectivity changes std::unordered_set all_screens; for( const SCH_SHEET_PATH& path : list ) all_screens.insert( path.LastScreen() ); SCH_RULE_AREA::UpdateRuleAreasInScreens( all_screens, aSchView ); // Recalculate all connectivity ConnectionGraph()->Recalculate( list, true, aChangedItemHandler, aProgressReporter ); } else { struct CHANGED_ITEM { SCH_ITEM* item; SCH_ITEM* linked_item; SCH_SCREEN* screen; }; // Final change sets std::set changed_items; std::set pts; std::set> item_paths; // Working change sets std::unordered_set changed_screens; std::set> changed_rule_areas; std::vector changed_connectable_items; // Lambda to add an item to the connectivity update sets auto addItemToChangeSet = [&changed_items, &pts, &item_paths]( CHANGED_ITEM itemData ) { std::vector& paths = itemData.screen->GetClientSheetPaths(); std::vector tmp_pts = itemData.item->GetConnectionPoints(); pts.insert( tmp_pts.begin(), tmp_pts.end() ); changed_items.insert( itemData.item ); for( SCH_SHEET_PATH& path : paths ) item_paths.insert( std::make_pair( path, itemData.item ) ); if( !itemData.linked_item || !itemData.linked_item->IsConnectable() ) return; tmp_pts = itemData.linked_item->GetConnectionPoints(); pts.insert( tmp_pts.begin(), tmp_pts.end() ); changed_items.insert( itemData.linked_item ); // We have to directly add the pins here because the link may not exist on the schematic // anymore and so won't be picked up by GetScreen()->Items().Overlapping() below. if( SCH_SYMBOL* symbol = dynamic_cast( itemData.linked_item ) ) { std::vector pins = symbol->GetPins(); changed_items.insert( pins.begin(), pins.end() ); } for( SCH_SHEET_PATH& path : paths ) item_paths.insert( std::make_pair( path, itemData.linked_item ) ); }; // Get all changed connectable items and determine all changed screens for( unsigned ii = 0; ii < aLastChangeList->GetCount(); ++ii ) { switch( aLastChangeList->GetPickedItemStatus( ii ) ) { // Only care about changed, new, and deleted items, the other // cases are not connectivity-related case UNDO_REDO::CHANGED: case UNDO_REDO::NEWITEM: case UNDO_REDO::DELETED: break; default: continue; } SCH_ITEM* item = dynamic_cast( aLastChangeList->GetPickedItem( ii ) ); if( item ) { SCH_SCREEN* screen = static_cast( aLastChangeList->GetScreenForItem( ii ) ); changed_screens.insert( screen ); if( item->Type() == SCH_RULE_AREA_T ) { SCH_RULE_AREA* ruleArea = static_cast( item ); changed_rule_areas.insert( { ruleArea, screen } ); } else if( item->IsConnectable() ) { SCH_ITEM* linked_item = dynamic_cast( aLastChangeList->GetPickedItemLink( ii ) ); changed_connectable_items.push_back( { item, linked_item, screen } ); } } } // Update rule areas in changed screens to propagate any directive connectivity changes std::vector> forceUpdateRuleAreas = SCH_RULE_AREA::UpdateRuleAreasInScreens( changed_screens, aSchView ); std::for_each( forceUpdateRuleAreas.begin(), forceUpdateRuleAreas.end(), [&]( std::pair& updatedRuleArea ) { changed_rule_areas.insert( updatedRuleArea ); } ); // If a SCH_RULE_AREA was changed, we need to add all past and present contained items to // update their connectivity std::map itemMap; list.FillItemMap( itemMap ); auto addPastAndPresentContainedItems = [&]( SCH_RULE_AREA* changedRuleArea, SCH_SCREEN* screen ) { for( const KIID& pastItem : changedRuleArea->GetPastContainedItems() ) { if( itemMap.contains( pastItem ) ) addItemToChangeSet( { static_cast( itemMap[pastItem] ), nullptr, screen } ); } for( SCH_ITEM* containedItem : changedRuleArea->GetContainedItems() ) addItemToChangeSet( { containedItem, nullptr, screen } ); }; for( const auto& [changedRuleArea, screen] : changed_rule_areas ) addPastAndPresentContainedItems( changedRuleArea, screen ); // Add all changed items, and associated items, to the change set for( CHANGED_ITEM& changed_item_data : changed_connectable_items ) { addItemToChangeSet( changed_item_data ); // If a SCH_DIRECTIVE_LABEL was changed which is attached to a SCH_RULE_AREA, we need // to add the contained items to the change set to force update of their connectivity if( changed_item_data.item->Type() == SCH_DIRECTIVE_LABEL_T ) { const std::vector labelConnectionPoints = changed_item_data.item->GetConnectionPoints(); auto candidateRuleAreas = changed_item_data.screen->Items().Overlapping( SCH_RULE_AREA_T, changed_item_data.item->GetBoundingBox() ); for( SCH_ITEM* candidateRuleArea : candidateRuleAreas ) { SCH_RULE_AREA* ruleArea = static_cast( candidateRuleArea ); std::vector borderShapes = ruleArea->MakeEffectiveShapes( true ); if( ruleArea->GetPolyShape().CollideEdge( labelConnectionPoints[0], nullptr, 5 ) ) addPastAndPresentContainedItems( ruleArea, changed_item_data.screen ); } } } for( const VECTOR2I& pt: pts ) { for( SCH_ITEM* item : GetCurrentScreen()->Items().Overlapping( pt ) ) { // Leave this check in place. Overlapping items are not necessarily connectable. if( !item->IsConnectable() ) continue; if( item->Type() == SCH_LINE_T ) { if( item->HitTest( pt ) ) changed_items.insert( item ); } else if( item->Type() == SCH_SYMBOL_T && item->IsConnected( pt ) ) { SCH_SYMBOL* symbol = static_cast( item ); std::vector pins = symbol->GetPins(); changed_items.insert( pins.begin(), pins.end() ); } else if( item->Type() == SCH_SHEET_T ) { SCH_SHEET* sheet = static_cast( item ); wxCHECK2( sheet, continue ); std::vector sheetPins = sheet->GetPins(); changed_items.insert( sheetPins.begin(), sheetPins.end() ); } else { if( item->IsConnected( pt ) ) changed_items.insert( item ); } } } std::set> all_items = ConnectionGraph()->ExtractAffectedItems( changed_items ); all_items.insert( item_paths.begin(), item_paths.end() ); CONNECTION_GRAPH new_graph( this ); new_graph.SetLastCodes( ConnectionGraph() ); std::shared_ptr netSettings = m_project->GetProjectFile().NetSettings(); std::set affectedNets; for( auto&[ path, item ] : all_items ) { wxCHECK2( item, continue ); item->SetConnectivityDirty(); SCH_CONNECTION* conn = item->Connection(); if( conn ) affectedNets.insert( conn->Name() ); } // Reset resolved netclass cache for this connection for( const wxString& netName : affectedNets ) netSettings->ClearCacheForNet( netName ); new_graph.Recalculate( list, false, aChangedItemHandler, aProgressReporter ); ConnectionGraph()->Merge( new_graph ); } if( !localCommit.Empty() ) localCommit.Push( _( "Schematic Cleanup" ) ); } void SCHEMATIC::CreateDefaultScreens() { Reset(); SCH_SHEET* rootSheet = new SCH_SHEET( this ); SetRoot( rootSheet ); SCH_SCREEN* rootScreen = new SCH_SCREEN( this ); const_cast( rootSheet->m_Uuid ) = rootScreen->GetUuid(); Root().SetScreen( rootScreen ); RootScreen()->SetFileName( wxEmptyString ); // Don't leave root page number empty SCH_SHEET_PATH rootSheetPath; rootSheetPath.push_back( rootSheet ); RootScreen()->SetPageNumber( wxT( "1" ) ); rootSheetPath.SetPageNumber( wxT( "1" ) ); // Rehash sheetpaths in hierarchy since we changed the uuid. RefreshHierarchy(); }