KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
sch_sheet_path.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2017 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright (C) 2011 Wayne Stambaugh <stambaughw@gmail.com>
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26#include <refdes_utils.h>
27#include <hash.h>
28#include <sch_screen.h>
29#include <sch_item.h>
30#include <sch_marker.h>
31#include <sch_label.h>
32#include <sch_reference_list.h>
33#include <symbol_library.h>
34#include <sch_sheet_path.h>
35#include <sch_symbol.h>
36#include <sch_sheet.h>
37#include <schematic.h>
38#include <template_fieldnames.h>
39#include <trace_helpers.h>
40
41#include <wx/filename.h>
42#include <wx/log.h>
43
44
51{
52public:
54 SCH_ITEM( nullptr, NOT_USED )
55 {}
56
57 wxString GetItemDescription( UNITS_PROVIDER* aUnitsProvider, bool aFull ) const override
58 {
59 return _( "(Deleted Item)" );
60 }
61
62 wxString GetClass() const override
63 {
64 return wxT( "DELETED_SHEET_ITEM" );
65 }
66
68 {
69 static DELETED_SHEET_ITEM* item = nullptr;
70
71 if( !item )
72 item = new DELETED_SHEET_ITEM();
73
74 return item;
75 }
76
77 // pure virtuals:
78 void SetPosition( const VECTOR2I& ) override {}
79 void Move( const VECTOR2I& aMoveVector ) override {}
80 void MirrorHorizontally( int aCenter ) override {}
81 void MirrorVertically( int aCenter ) override {}
82 void Rotate( const VECTOR2I& aCenter, bool aRotateCCW ) override {}
83
84 double Similarity( const SCH_ITEM& aOther ) const override
85 {
86 return 0.0;
87 }
88
89 bool operator==( const SCH_ITEM& aOther ) const override
90 {
91 return false;
92 }
93
94#if defined(DEBUG)
95 void Show( int , std::ostream& ) const override {}
96#endif
97};
98
99
100namespace std
101{
102 size_t hash<SCH_SHEET_PATH>::operator()( const SCH_SHEET_PATH& path ) const
103 {
104 return path.GetCurrentHash();
105 }
106}
107
108
110{
112 m_current_hash = 0;
113}
114
115
117{
118 initFromOther( aOther );
119}
120
121
123{
124 initFromOther( aOther );
125 return *this;
126}
127
128
130{
131 SCH_SHEET_PATH retv = *this;
132
133 size_t size = aOther.size();
134
135 for( size_t i = 0; i < size; i++ )
136 retv.push_back( aOther.at( i ) );
137
138 return retv;
139}
140
141
143{
144 m_sheets = aOther.m_sheets;
148
149 // Note: don't copy m_recursion_test_cache as it is slow and we want
150 // std::vector<SCH_SHEET_PATH> to be very fast to construct for use in
151 // the connectivity algorithm.
152}
153
154
156{
157 // The root sheet path is empty. All other sheet paths must start with the root sheet path.
158 return ( m_sheets.size() == 0 ) || ( GetSheet( 0 )->IsRootSheet() );
159}
160
161
163{
164 m_current_hash = 0;
165
166 for( SCH_SHEET* sheet : m_sheets )
167 hash_combine( m_current_hash, sheet->m_Uuid.Hash() );
168}
169
170
171int SCH_SHEET_PATH::Cmp( const SCH_SHEET_PATH& aSheetPathToTest ) const
172{
173 if( size() > aSheetPathToTest.size() )
174 return 1;
175
176 if( size() < aSheetPathToTest.size() )
177 return -1;
178
179 // otherwise, same number of sheets.
180 for( unsigned i = 0; i < size(); i++ )
181 {
182 if( at( i )->m_Uuid < aSheetPathToTest.at( i )->m_Uuid )
183 return -1;
184
185 if( at( i )->m_Uuid != aSheetPathToTest.at( i )->m_Uuid )
186 return 1;
187 }
188
189 return 0;
190}
191
192
193int SCH_SHEET_PATH::ComparePageNum( const SCH_SHEET_PATH& aSheetPathToTest ) const
194{
195 wxString pageA = this->GetPageNumber();
196 wxString pageB = aSheetPathToTest.GetPageNumber();
197
198 int pageNumComp = SCH_SHEET::ComparePageNum( pageA, pageB );
199
200 if( pageNumComp == 0 )
201 {
202 int virtualPageA = GetVirtualPageNumber();
203 int virtualPageB = aSheetPathToTest.GetVirtualPageNumber();
204
205 if( virtualPageA > virtualPageB )
206 pageNumComp = 1;
207 else if( virtualPageA < virtualPageB )
208 pageNumComp = -1;
209 }
210
211 return pageNumComp;
212}
213
214
215bool SCH_SHEET_PATH::IsContainedWithin( const SCH_SHEET_PATH& aSheetPathToTest ) const
216{
217 if( aSheetPathToTest.size() > size() )
218 return false;
219
220 for( size_t i = 0; i < aSheetPathToTest.size(); ++i )
221 {
222 if( at( i )->m_Uuid != aSheetPathToTest.at( i )->m_Uuid )
223 {
224 wxLogTrace( traceSchSheetPaths, "Sheet path '%s' is not within path '%s'.",
225 aSheetPathToTest.Path().AsString(), Path().AsString() );
226
227 return false;
228 }
229 }
230
231 wxLogTrace( traceSchSheetPaths, "Sheet path '%s' is within path '%s'.",
232 aSheetPathToTest.Path().AsString(), Path().AsString() );
233
234 return true;
235}
236
237
239{
240 if( !empty() )
241 return m_sheets.back();
242
243 return nullptr;
244}
245
246
248{
249 SCH_SHEET* lastSheet = Last();
250
251 if( lastSheet )
252 return lastSheet->GetScreen();
253
254 return nullptr;
255}
256
257
259{
260 SCH_SHEET* lastSheet = Last();
261
262 if( lastSheet )
263 return lastSheet->GetScreen();
264
265 return nullptr;
266}
267
268
270{
271 for( SCH_SHEET* sheet : m_sheets )
272 {
273 if( sheet->GetExcludedFromSim() )
274 return true;
275 }
276
277 return false;
278}
279
280
282{
283 for( SCH_SHEET* sheet : m_sheets )
284 {
285 if( sheet->GetExcludedFromBOM() )
286 return true;
287 }
288
289 return false;
290}
291
292
294{
295 for( SCH_SHEET* sheet : m_sheets )
296 {
297 if( sheet->GetExcludedFromBoard() )
298 return true;
299 }
300
301 return false;
302}
303
304
306{
307 for( SCH_SHEET* sheet : m_sheets )
308 {
309 if( sheet->GetDNP() )
310 return true;
311 }
312
313 return false;
314}
315
316
318{
319 wxString s;
320
321 s = wxT( "/" ); // This is the root path
322
323 // Start at 1 to avoid the root sheet, which does not need to be added to the path.
324 // Its timestamp changes anyway.
325 for( unsigned i = 1; i < size(); i++ )
326 s += at( i )->m_Uuid.AsString() + "/";
327
328 return s;
329}
330
331
333{
335 path.reserve( m_sheets.size() );
336
337 for( const SCH_SHEET* sheet : m_sheets )
338 path.push_back( sheet->m_Uuid );
339
340 return path;
341}
342
343
344wxString SCH_SHEET_PATH::PathHumanReadable( bool aUseShortRootName,
345 bool aStripTrailingSeparator ) const
346{
347 wxString s;
348
349 if( aUseShortRootName )
350 {
351 s = wxS( "/" ); // Use only the short name in netlists
352 }
353 else
354 {
355 wxString fileName;
356
357 if( !empty() && at( 0 )->GetScreen() )
358 fileName = at( 0 )->GetScreen()->GetFileName();
359
360 wxFileName fn = fileName;
361
362 s = fn.GetName() + wxS( "/" );
363 }
364
365 // Start at 1 since we've already processed the root sheet.
366 for( unsigned i = 1; i < size(); i++ )
367 s << at( i )->GetField( FIELD_T::SHEET_NAME )->GetShownText( false ) << wxS( "/" );
368
369 if( aStripTrailingSeparator && s.EndsWith( "/" ) )
370 s = s.Left( s.length() - 1 );
371
372 return s;
373}
374
375
377{
378 std::vector<SCH_ITEM*> items;
379
380 std::copy_if( LastScreen()->Items().begin(), LastScreen()->Items().end(),
381 std::back_inserter( items ),
382 []( SCH_ITEM* aItem )
383 {
384 return ( aItem->Type() == SCH_SYMBOL_T || aItem->Type() == SCH_GLOBAL_LABEL_T );
385 } );
386
387 for( SCH_ITEM* item : items )
388 {
389 if( item->Type() == SCH_SYMBOL_T )
390 {
391 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
392
393 symbol->GetField( FIELD_T::REFERENCE )->SetText( symbol->GetRef( this ) );
394 symbol->SetUnit( symbol->GetUnitSelection( this ) );
395 LastScreen()->Update( item, false );
396 }
397 else if( item->Type() == SCH_GLOBAL_LABEL_T )
398 {
399 SCH_GLOBALLABEL* label = static_cast<SCH_GLOBALLABEL*>( item );
400
401 if( label->GetFields().size() > 0 ) // Possible when reading a legacy .sch schematic
402 {
403 SCH_FIELD* intersheetRefs = label->GetField( FIELD_T::INTERSHEET_REFS );
404
405 // Fixup for legacy files which didn't store a position for the intersheet refs
406 // unless they were shown.
407 if( intersheetRefs->GetPosition() == VECTOR2I() && !intersheetRefs->IsVisible() )
409
410 intersheetRefs->SetVisible( label->Schematic()->Settings().m_IntersheetRefsShow );
411 LastScreen()->Update( intersheetRefs );
412 }
413 }
414 }
415}
416
417
418void SCH_SHEET_PATH::GetSymbols( SCH_REFERENCE_LIST& aReferences, bool aIncludePowerSymbols,
419 bool aForceIncludeOrphanSymbols ) const
420{
421 for( SCH_ITEM* item : LastScreen()->Items().OfType( SCH_SYMBOL_T ) )
422 {
423 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
424 AppendSymbol( aReferences, symbol, aIncludePowerSymbols, aForceIncludeOrphanSymbols );
425 }
426}
427
428
430 bool aIncludePowerSymbols,
431 bool aForceIncludeOrphanSymbols ) const
432{
433 // Skip pseudo-symbols, which have a reference starting with #. This mainly
434 // affects power symbols.
435 if( aIncludePowerSymbols || aSymbol->GetRef( this )[0] != wxT( '#' ) )
436 {
437 if( aSymbol->GetLibSymbolRef() || aForceIncludeOrphanSymbols )
438 {
439 SCH_REFERENCE schReference( aSymbol, *this );
440
441 schReference.SetSheetNumber( m_virtualPageNumber );
442 aReferences.AddItem( schReference );
443 }
444 }
445}
446
447
449 bool aIncludePowerSymbols ) const
450{
451 for( SCH_ITEM* item : LastScreen()->Items().OfType( SCH_SYMBOL_T ) )
452 {
453 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
454 AppendMultiUnitSymbol( aRefList, symbol, aIncludePowerSymbols );
455 }
456}
457
458
460 SCH_SYMBOL* aSymbol,
461 bool aIncludePowerSymbols ) const
462{
463 // Skip pseudo-symbols, which have a reference starting with #. This mainly
464 // affects power symbols.
465 if( !aIncludePowerSymbols && aSymbol->GetRef( this )[0] == wxT( '#' ) )
466 return;
467
468 LIB_SYMBOL* symbol = aSymbol->GetLibSymbolRef().get();
469
470 if( symbol && symbol->GetUnitCount() > 1 )
471 {
472 SCH_REFERENCE schReference = SCH_REFERENCE( aSymbol, *this );
473 schReference.SetSheetNumber( m_virtualPageNumber );
474 wxString reference_str = schReference.GetRef();
475
476 // Never lock unassigned references
477 if( reference_str[reference_str.Len() - 1] == '?' )
478 return;
479
480 aRefList[reference_str].AddItem( schReference );
481 }
482}
483
484
486{
487 return m_current_hash == d1.GetCurrentHash();
488}
489
490
491bool SCH_SHEET_PATH::TestForRecursion( const wxString& aSrcFileName, const wxString& aDestFileName )
492{
493 auto pair = std::make_pair( aSrcFileName, aDestFileName );
494
495 if( m_recursion_test_cache.count( pair ) )
496 return m_recursion_test_cache.at( pair );
497
498 SCHEMATIC* sch = LastScreen()->Schematic();
499
500 wxCHECK_MSG( sch, false, "No SCHEMATIC found in SCH_SHEET_PATH::TestForRecursion!" );
501
502 wxFileName rootFn = sch->GetFileName();
503 wxFileName srcFn = aSrcFileName;
504 wxFileName destFn = aDestFileName;
505
506 if( srcFn.IsRelative() )
507 srcFn.MakeAbsolute( rootFn.GetPath() );
508
509 if( destFn.IsRelative() )
510 destFn.MakeAbsolute( rootFn.GetPath() );
511
512 // The source and destination sheet file names cannot be the same.
513 if( srcFn == destFn )
514 {
515 m_recursion_test_cache[pair] = true;
516 return true;
517 }
518
522 unsigned i = 0;
523
524 while( i < size() )
525 {
526 wxFileName cmpFn = at( i )->GetFileName();
527
528 if( cmpFn.IsRelative() )
529 cmpFn.MakeAbsolute( rootFn.GetPath() );
530
531 // Test if the file name of the destination sheet is in anywhere in this sheet path.
532 if( cmpFn == destFn )
533 break;
534
535 i++;
536 }
537
538 // The destination sheet file name was not found in the sheet path or the destination
539 // sheet file name is the root sheet so no recursion is possible.
540 if( i >= size() || i == 0 )
541 {
542 m_recursion_test_cache[pair] = false;
543 return false;
544 }
545
546 // Walk back up to the root sheet to see if the source file name is already a parent in
547 // the sheet path. If so, recursion will occur.
548 do
549 {
550 i -= 1;
551
552 wxFileName cmpFn = at( i )->GetFileName();
553
554 if( cmpFn.IsRelative() )
555 cmpFn.MakeAbsolute( rootFn.GetPath() );
556
557 if( cmpFn == srcFn )
558 {
559 m_recursion_test_cache[pair] = true;
560 return true;
561 }
562
563 } while( i != 0 );
564
565 // The source sheet file name is not a parent of the destination sheet file name.
566 m_recursion_test_cache[pair] = false;
567 return false;
568}
569
570
572{
573 SCH_SHEET* sheet = Last();
574
575 wxCHECK( sheet, wxEmptyString );
576
577 KIID_PATH tmpPath = Path();
578 tmpPath.pop_back();
579
580 return sheet->getPageNumber( tmpPath );
581}
582
583
584void SCH_SHEET_PATH::SetPageNumber( const wxString& aPageNumber )
585{
586 SCH_SHEET* sheet = Last();
587
588 wxCHECK( sheet, /* void */ );
589
590 KIID_PATH tmpPath = Path();
591
592 tmpPath.pop_back();
593
594 sheet->addInstance( tmpPath );
595 sheet->setPageNumber( tmpPath, aPageNumber );
596}
597
598
600 const wxString& aProjectName )
601{
602 wxCHECK( !aProjectName.IsEmpty(), /* void */ );
603
604 SCH_SHEET_PATH newSheetPath( aPrefixSheetPath );
605 SCH_SHEET_PATH currentSheetPath( *this );
606
607 // Prefix the new hierarchical path.
608 newSheetPath = newSheetPath + currentSheetPath;
609
610 for( SCH_ITEM* item : LastScreen()->Items().OfType( SCH_SYMBOL_T ) )
611 {
612 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
613
614 wxCHECK2( symbol, continue );
615
616 SCH_SYMBOL_INSTANCE newSymbolInstance;
617
618 if( symbol->GetInstance( newSymbolInstance, Path(), true ) )
619 {
620 newSymbolInstance.m_ProjectName = aProjectName;
621
622 // Use an existing symbol instance for this path if it exists.
623 newSymbolInstance.m_Path = newSheetPath.Path();
624 symbol->AddHierarchicalReference( newSymbolInstance );
625 }
626 else if( !symbol->GetInstances().empty() )
627 {
628 newSymbolInstance.m_ProjectName = aProjectName;
629
630 // Use the first symbol instance if any symbol instance data exists.
631 newSymbolInstance = symbol->GetInstances()[0];
632 newSymbolInstance.m_Path = newSheetPath.Path();
633 symbol->AddHierarchicalReference( newSymbolInstance );
634 }
635 else
636 {
637 newSymbolInstance.m_ProjectName = aProjectName;
638
639 // Fall back to the last saved symbol field and unit settings if there is no
640 // instance data.
641 newSymbolInstance.m_Path = newSheetPath.Path();
642 newSymbolInstance.m_Reference = symbol->GetField( FIELD_T::REFERENCE )->GetText();
643 newSymbolInstance.m_Unit = symbol->GetUnit();
644 symbol->AddHierarchicalReference( newSymbolInstance );
645 }
646 }
647}
648
649
651{
652 for( SCH_ITEM* item : LastScreen()->Items().OfType( SCH_SYMBOL_T ) )
653 {
654 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
655
656 wxCHECK2( symbol, continue );
657
658 SCH_SHEET_PATH fullSheetPath( aPrefixSheetPath );
659 SCH_SHEET_PATH currentSheetPath( *this );
660
661 // Prefix the hierarchical path of the symbol instance to be removed.
662 fullSheetPath = fullSheetPath + currentSheetPath;
663 symbol->RemoveInstance( fullSheetPath );
664 }
665}
666
667
668void SCH_SHEET_PATH::CheckForMissingSymbolInstances( const wxString& aProjectName )
669{
670 wxCHECK( !aProjectName.IsEmpty() && LastScreen(), /* void */ );
671
672 for( SCH_ITEM* item : LastScreen()->Items().OfType( SCH_SYMBOL_T ) )
673 {
674 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
675
676 wxCHECK2( symbol, continue );
677
678 SCH_SYMBOL_INSTANCE symbolInstance;
679
680 if( !symbol->GetInstance( symbolInstance, Path() ) )
681 {
682 wxLogTrace( traceSchSheetPaths, "Adding missing symbol \"%s\" instance data for "
683 "sheet path '%s'.",
684 symbol->m_Uuid.AsString(), PathHumanReadable( false ) );
685
686 // Legacy schematics that are not shared do not contain separate instance data.
687 // The symbol reference and unit are saved in the reference field and unit entries.
688 if( ( LastScreen()->GetRefCount() <= 1 ) &&
689 ( LastScreen()->GetFileFormatVersionAtLoad() <= 20200310 ) )
690 {
691 SCH_FIELD* refField = symbol->GetField( FIELD_T::REFERENCE );
692 symbolInstance.m_Reference = refField->GetShownText( this, true );
693 symbolInstance.m_Unit = symbol->GetUnit();
694 }
695 else
696 {
697 // When schematics are shared, we cannot know which instance the current symbol
698 // reference field and unit belong to. In this case, we clear the reference
699 // annotation and set the unit to 1.
700 symbolInstance.m_Reference = UTIL::GetRefDesUnannotated( symbol->GetPrefix() );
701 }
702
703 symbolInstance.m_ProjectName = aProjectName;
704 symbolInstance.m_Path = Path();
705 symbol->AddHierarchicalReference( symbolInstance );
706 }
707 }
708}
709
710
712{
713 wxCHECK( m_sheets.size() > 1, /* void */ );
714
715 wxFileName sheetFileName = Last()->GetFileName();
716
717 // If the sheet file name is absolute, then the user requested is so don't make it relative.
718 if( sheetFileName.IsAbsolute() )
719 return;
720
721 SCH_SCREEN* screen = LastScreen();
722 SCH_SCREEN* parentScreen = m_sheets[ m_sheets.size() - 2 ]->GetScreen();
723
724 wxCHECK( screen && parentScreen, /* void */ );
725
726 wxFileName fileName = screen->GetFileName();
727 wxFileName parentFileName = parentScreen->GetFileName();
728
729 // SCH_SCREEN file names must be absolute. If they are not, someone set them incorrectly
730 // on load or on creation.
731 wxCHECK( fileName.IsAbsolute() && parentFileName.IsAbsolute(), /* void */ );
732
733 if( fileName.GetPath() == parentFileName.GetPath() )
734 {
735 Last()->SetFileName( fileName.GetFullName() );
736 }
737 else if( fileName.MakeRelativeTo( parentFileName.GetPath() ) )
738 {
739 Last()->SetFileName( fileName.GetFullPath() );
740 }
741 else
742 {
743 Last()->SetFileName( screen->GetFileName() );
744 }
745
746 wxLogTrace( tracePathsAndFiles,
747 wxT( "\n File name: '%s'"
748 "\n parent file name '%s',"
749 "\n sheet '%s' file name '%s'." ),
750 screen->GetFileName(), parentScreen->GetFileName(), PathHumanReadable(),
751 Last()->GetFileName() );
752}
753
754
756{
757 if( aSheet != nullptr )
758 BuildSheetList( aSheet, false );
759}
760
761
762void SCH_SHEET_LIST::BuildSheetList( SCH_SHEET* aSheet, bool aCheckIntegrity )
763{
764 if( !aSheet )
765 return;
766
767 std::vector<SCH_SHEET*> badSheets;
768
770 m_currentSheetPath.SetVirtualPageNumber( static_cast<int>( size() ) + 1 );
771 push_back( m_currentSheetPath );
772
774 {
775 wxString parentFileName = aSheet->GetFileName();
776 std::vector<SCH_ITEM*> childSheets;
777 m_currentSheetPath.LastScreen()->GetSheets( &childSheets );
778
779 for( SCH_ITEM* item : childSheets )
780 {
781 SCH_SHEET* sheet = static_cast<SCH_SHEET*>( item );
782
783 if( aCheckIntegrity )
784 {
785 if( !m_currentSheetPath.TestForRecursion( sheet->GetFileName(), parentFileName ) )
786 BuildSheetList( sheet, true );
787 else
788 badSheets.push_back( sheet );
789 }
790 else
791 {
792 // If we are not performing a full recursion test, at least check if we are in
793 // a simple recursion scenario to prevent stack overflow crashes
794 wxCHECK2_MSG( sheet->GetFileName() != aSheet->GetFileName(), continue,
795 wxT( "Recursion prevented in SCH_SHEET_LIST::BuildSheetList" ) );
796
797 BuildSheetList( sheet, false );
798 }
799 }
800 }
801
802 if( aCheckIntegrity )
803 {
804 for( SCH_SHEET* sheet : badSheets )
805 {
808 }
809 }
810
812}
813
814
815void SCH_SHEET_LIST::SortByHierarchicalPageNumbers( bool aUpdateVirtualPageNums )
816{
817 for( const SCH_SHEET_PATH& path : *this )
818 path.CachePageNumber();
819
820 std::sort( begin(), end(),
821 []( const SCH_SHEET_PATH& a, const SCH_SHEET_PATH& b ) -> bool
822 {
823 if( a.size() != b.size() )
824 return a.size() < b.size();
825
828
829 if( retval < 0 )
830 return true;
831 else if( retval > 0 )
832 return false;
833
835 return true;
836 else if( a.GetVirtualPageNumber() > b.GetVirtualPageNumber() )
837 return false;
838
839 // Enforce strict ordering. If the page numbers are the same, use UUIDs
840 return a.GetCurrentHash() < b.GetCurrentHash();
841 } );
842
843 if( aUpdateVirtualPageNums )
844 {
845 int virtualPageNum = 1;
846
847 for( SCH_SHEET_PATH& sheet : *this )
848 sheet.SetVirtualPageNumber( virtualPageNum++ );
849 }
850}
851
852
853void SCH_SHEET_LIST::SortByPageNumbers( bool aUpdateVirtualPageNums )
854{
855 for( const SCH_SHEET_PATH& path : *this )
856 path.CachePageNumber();
857
858 std::sort( begin(), end(),
859 []( const SCH_SHEET_PATH& a, const SCH_SHEET_PATH& b ) -> bool
860 {
863
864 if( retval < 0 )
865 return true;
866 else if( retval > 0 )
867 return false;
868
870 return true;
871 else if( a.GetVirtualPageNumber() > b.GetVirtualPageNumber() )
872 return false;
873
874 // Enforce strict ordering. If the page numbers are the same, use UUIDs
875 return a.GetCurrentHash() < b.GetCurrentHash();
876 } );
877
878 if( aUpdateVirtualPageNums )
879 {
880 int virtualPageNum = 1;
881
882 for( SCH_SHEET_PATH& sheet : *this )
883 sheet.SetVirtualPageNumber( virtualPageNum++ );
884 }
885}
886
887
888bool SCH_SHEET_LIST::NameExists( const wxString& aSheetName ) const
889{
890 for( const SCH_SHEET_PATH& sheet : *this )
891 {
892 if( sheet.Last()->GetName() == aSheetName )
893 return true;
894 }
895
896 return false;
897}
898
899
900bool SCH_SHEET_LIST::PageNumberExists( const wxString& aPageNumber ) const
901{
902 for( const SCH_SHEET_PATH& sheet : *this )
903 {
904 if( sheet.GetPageNumber() == aPageNumber )
905 return true;
906 }
907
908 return false;
909}
910
911
912void SCH_SHEET_LIST::TrimToPageNumbers( const std::vector<wxString>& aPageInclusions )
913{
914 auto it = std::remove_if( begin(), end(),
915 [&]( SCH_SHEET_PATH sheet )
916 {
917 return std::find( aPageInclusions.begin(), aPageInclusions.end(),
918 sheet.GetPageNumber() )
919 == aPageInclusions.end();
920 } );
921
922 erase( it, end() );
923}
924
925
927{
928 for( const SCH_SHEET_PATH& sheet : *this )
929 {
930 if( sheet.LastScreen() && sheet.LastScreen()->IsContentModified() )
931 return true;
932 }
933
934 return false;
935}
936
937
939{
940 for( const SCH_SHEET_PATH& sheet : *this )
941 {
942 if( sheet.LastScreen() )
943 sheet.LastScreen()->SetContentModified( false );
944 }
945}
946
947
949{
950 for( const SCH_SHEET_PATH& sheet : *this )
951 {
952 SCH_ITEM* item = sheet.GetItem( aID );
953
954 if( item )
955 {
956 if( aPathOut )
957 *aPathOut = sheet;
958
959 return item;
960 }
961 }
962
963 // Not found; weak reference has been deleted.
965}
966
967
969{
970 for( SCH_ITEM* aItem : LastScreen()->Items() )
971 {
972 if( aItem->m_Uuid == aID )
973 return aItem;
974
975 SCH_ITEM* childMatch = nullptr;
976
977 aItem->RunOnChildren(
978 [&]( SCH_ITEM* aChild )
979 {
980 if( aChild->m_Uuid == aID )
981 childMatch = aChild;
982 },
983 RECURSE_MODE::NO_RECURSE );
984
985 if( childMatch )
986 return childMatch;
987 }
988
989 return nullptr;
990}
991
992
993void SCH_SHEET_LIST::FillItemMap( std::map<KIID, EDA_ITEM*>& aMap )
994{
995 for( const SCH_SHEET_PATH& sheet : *this )
996 {
997 SCH_SCREEN* screen = sheet.LastScreen();
998
999 for( SCH_ITEM* aItem : screen->Items() )
1000 {
1001 aMap[ aItem->m_Uuid ] = aItem;
1002
1003 aItem->RunOnChildren(
1004 [&]( SCH_ITEM* aChild )
1005 {
1006 aMap[ aChild->m_Uuid ] = aChild;
1007 },
1008 RECURSE_MODE::NO_RECURSE );
1009 }
1010 }
1011}
1012
1013
1015{
1016 // List of reference for power symbols
1017 SCH_REFERENCE_LIST references;
1018 SCH_REFERENCE_LIST additionalreferences; // Todo: add as a parameter to this function
1019
1020 // Map of locked symbols (not used, but needed by Annotate()
1021 SCH_MULTI_UNIT_REFERENCE_MAP lockedSymbols;
1022
1023 // Build the list of power symbols:
1024 for( SCH_SHEET_PATH& sheet : *this )
1025 {
1026 for( SCH_ITEM* item : sheet.LastScreen()->Items().OfType( SCH_SYMBOL_T ) )
1027 {
1028 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
1029 LIB_SYMBOL* libSymbol = symbol->GetLibSymbolRef().get();
1030
1031 if( libSymbol && libSymbol->IsPower() )
1032 {
1033 SCH_REFERENCE schReference( symbol, sheet );
1034 references.AddItem( schReference );
1035 }
1036 }
1037 }
1038
1039 // Find duplicate, and silently clear annotation of duplicate
1040 std::map<wxString, int> ref_list; // stores the existing references
1041
1042 for( unsigned ii = 0; ii< references.GetCount(); ++ii )
1043 {
1044 wxString curr_ref = references[ii].GetRef();
1045
1046 if( ref_list.find( curr_ref ) == ref_list.end() )
1047 {
1048 ref_list[curr_ref] = ii;
1049 continue;
1050 }
1051
1052 // Possible duplicate, if the ref ends by a number:
1053 if( curr_ref.Last() < '0' && curr_ref.Last() > '9' )
1054 continue; // not annotated
1055
1056 // Duplicate: clear annotation by removing the number ending the ref
1057 while( curr_ref.Last() >= '0' && curr_ref.Last() <= '9' )
1058 curr_ref.RemoveLast();
1059
1060 references[ii].SetRef( curr_ref );
1061 }
1062
1063 // Break full symbol reference into name (prefix) and number:
1064 // example: IC1 become IC, and 1
1065 references.SplitReferences();
1066
1067 // Ensure all power symbols have the reference starting by '#'
1068 // (Not sure this is really useful)
1069 for( unsigned ii = 0; ii< references.GetCount(); ++ii )
1070 {
1071 if( references[ii].GetRef()[0] != '#' )
1072 {
1073 wxString new_ref = "#" + references[ii].GetRef();
1074 references[ii].SetRef( new_ref );
1075 }
1076 }
1077
1078 // Recalculate and update reference numbers in schematic
1079 references.Annotate( false, 0, 100, lockedSymbols, additionalreferences );
1080 references.UpdateAnnotation();
1081}
1082
1083
1084void SCH_SHEET_LIST::GetSymbols( SCH_REFERENCE_LIST& aReferences, bool aIncludePowerSymbols,
1085 bool aForceIncludeOrphanSymbols ) const
1086{
1087 for( const SCH_SHEET_PATH& sheet : *this )
1088 sheet.GetSymbols( aReferences, aIncludePowerSymbols, aForceIncludeOrphanSymbols );
1089}
1090
1091
1093 const SCH_SHEET_PATH& aSheetPath,
1094 bool aIncludePowerSymbols,
1095 bool aForceIncludeOrphanSymbols ) const
1096{
1097 for( const SCH_SHEET_PATH& sheet : *this )
1098 {
1099 if( sheet.IsContainedWithin( aSheetPath ) )
1100 sheet.GetSymbols( aReferences, aIncludePowerSymbols, aForceIncludeOrphanSymbols );
1101 }
1102}
1103
1104
1105void SCH_SHEET_LIST::GetSheetsWithinPath( std::vector<SCH_SHEET_PATH>& aSheets,
1106 const SCH_SHEET_PATH& aSheetPath ) const
1107{
1108 for( const SCH_SHEET_PATH& sheet : *this )
1109 {
1110 if( sheet.IsContainedWithin( aSheetPath ) )
1111 aSheets.push_back( sheet );
1112 }
1113}
1114
1115
1116std::optional<SCH_SHEET_PATH> SCH_SHEET_LIST::GetSheetPathByKIIDPath( const KIID_PATH& aPath,
1117 bool aIncludeLastSheet ) const
1118{
1119 for( const SCH_SHEET_PATH& sheet : *this )
1120 {
1121 KIID_PATH testPath = sheet.Path();
1122
1123 if( !aIncludeLastSheet )
1124 testPath.pop_back();
1125
1126 if( testPath == aPath )
1127 return SCH_SHEET_PATH( sheet );
1128 }
1129
1130 return std::nullopt;
1131}
1132
1133
1135 bool aIncludePowerSymbols ) const
1136{
1137 for( auto it = begin(); it != end(); ++it )
1138 {
1140 ( *it ).GetMultiUnitSymbols( tempMap, aIncludePowerSymbols );
1141
1142 for( SCH_MULTI_UNIT_REFERENCE_MAP::value_type& pair : tempMap )
1143 {
1144 // Merge this list into the main one
1145 unsigned n_refs = pair.second.GetCount();
1146
1147 for( unsigned thisRef = 0; thisRef < n_refs; ++thisRef )
1148 aRefList[pair.first].AddItem( pair.second[thisRef] );
1149 }
1150 }
1151}
1152
1153
1154bool SCH_SHEET_LIST::TestForRecursion( const SCH_SHEET_LIST& aSrcSheetHierarchy,
1155 const wxString& aDestFileName )
1156{
1157 if( empty() )
1158 return false;
1159
1160 SCHEMATIC* sch = at( 0 ).LastScreen()->Schematic();
1161
1162 wxCHECK_MSG( sch, false, "No SCHEMATIC found in SCH_SHEET_LIST::TestForRecursion!" );
1163
1164 wxFileName rootFn = sch->GetFileName();
1165 wxFileName destFn = aDestFileName;
1166
1167 if( destFn.IsRelative() )
1168 destFn.MakeAbsolute( rootFn.GetPath() );
1169
1170 // Test each SCH_SHEET_PATH in this SCH_SHEET_LIST for potential recursion.
1171 for( unsigned i = 0; i < size(); i++ )
1172 {
1173 // Test each SCH_SHEET_PATH in the source sheet.
1174 for( unsigned j = 0; j < aSrcSheetHierarchy.size(); j++ )
1175 {
1176 const SCH_SHEET_PATH* sheetPath = &aSrcSheetHierarchy[j];
1177
1178 for( unsigned k = 0; k < sheetPath->size(); k++ )
1179 {
1180 if( at( i ).TestForRecursion( sheetPath->GetSheet( k )->GetFileName(),
1181 aDestFileName ) )
1182 {
1183 return true;
1184 }
1185 }
1186 }
1187 }
1188
1189 // The source sheet file can safely be added to the destination sheet file.
1190 return false;
1191}
1192
1193
1195{
1196 for( SCH_SHEET_PATH& path : *this )
1197 {
1198 if( path.Path() == aPath->Path() )
1199 return &path;
1200 }
1201
1202 return nullptr;
1203}
1204
1205
1207{
1208 for( SCH_SHEET_PATH& sheetpath : *this )
1209 {
1210 if( sheetpath.LastScreen() == aScreen )
1211 return sheetpath;
1212 }
1213
1214 return SCH_SHEET_PATH();
1215}
1216
1217
1219{
1220 SCH_SHEET_LIST retval;
1221
1222 for( const SCH_SHEET_PATH& sheetpath : *this )
1223 {
1224 if( sheetpath.LastScreen() == aScreen )
1225 retval.push_back( sheetpath );
1226 }
1227
1228 return retval;
1229}
1230
1231
1233 const std::vector<SCH_SYMBOL_INSTANCE>& aSymbolInstances )
1234{
1235 for( SCH_SHEET_PATH& sheetPath : *this )
1236 {
1237 for( SCH_ITEM* item : sheetPath.LastScreen()->Items().OfType( SCH_SYMBOL_T ) )
1238 {
1239 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
1240
1241 wxCHECK2( symbol, continue );
1242
1243 KIID_PATH sheetPathWithSymbolUuid = sheetPath.Path();
1244 sheetPathWithSymbolUuid.push_back( symbol->m_Uuid );
1245
1246 auto it = std::find_if( aSymbolInstances.begin(), aSymbolInstances.end(),
1247 [ sheetPathWithSymbolUuid ]( const SCH_SYMBOL_INSTANCE& r ) -> bool
1248 {
1249 return sheetPathWithSymbolUuid == r.m_Path;
1250 } );
1251
1252 if( it == aSymbolInstances.end() )
1253 {
1254 wxLogTrace( traceSchSheetPaths, "No symbol instance found for symbol '%s'",
1255 sheetPathWithSymbolUuid.AsString() );
1256 continue;
1257 }
1258
1259 // Symbol instance paths are stored and looked up in memory with the root path so use
1260 // the full path here.
1261 symbol->AddHierarchicalReference( sheetPath.Path(), it->m_Reference, it->m_Unit );
1262 symbol->GetField( FIELD_T::REFERENCE )->SetText( it->m_Reference );
1263
1264 if( !it->m_Value.IsEmpty() )
1265 symbol->SetValueFieldText( it->m_Value );
1266
1267 if( !it->m_Footprint.IsEmpty() )
1268 symbol->SetFootprintFieldText( it->m_Footprint );
1269
1270 symbol->UpdatePrefix();
1271 }
1272 }
1273}
1274
1275
1276void SCH_SHEET_LIST::UpdateSheetInstanceData( const std::vector<SCH_SHEET_INSTANCE>& aSheetInstances )
1277{
1278
1279 for( SCH_SHEET_PATH& path : *this )
1280 {
1281 SCH_SHEET* sheet = path.Last();
1282
1283 wxCHECK2( sheet && path.Last(), continue );
1284
1285 auto it = std::find_if( aSheetInstances.begin(), aSheetInstances.end(),
1286 [ path ]( const SCH_SHEET_INSTANCE& r ) -> bool
1287 {
1288 return path.Path() == r.m_Path;
1289 } );
1290
1291 if( it == aSheetInstances.end() )
1292 {
1293 wxLogTrace( traceSchSheetPaths, "No sheet instance found for path '%s'",
1294 path.Path().AsString() );
1295 continue;
1296 }
1297
1298 wxLogTrace( traceSchSheetPaths, "Setting sheet '%s' instance '%s' page number '%s'",
1299 ( sheet->GetName().IsEmpty() ) ? wxString( wxT( "root" ) ) : sheet->GetName(),
1300 path.Path().AsString(), it->m_PageNumber );
1301 path.SetPageNumber( it->m_PageNumber );
1302 }
1303}
1304
1305
1306std::vector<KIID_PATH> SCH_SHEET_LIST::GetPaths() const
1307{
1308 std::vector<KIID_PATH> paths;
1309
1310 for( const SCH_SHEET_PATH& sheetPath : *this )
1311 paths.emplace_back( sheetPath.Path() );
1312
1313 return paths;
1314}
1315
1316
1317std::vector<SCH_SHEET_INSTANCE> SCH_SHEET_LIST::GetSheetInstances() const
1318{
1319 std::vector<SCH_SHEET_INSTANCE> retval;
1320
1321 for( const SCH_SHEET_PATH& path : *this )
1322 {
1323 const SCH_SHEET* sheet = path.Last();
1324
1325 wxCHECK2( sheet, continue );
1326
1327 SCH_SHEET_INSTANCE instance;
1328 SCH_SHEET_PATH tmpPath = path;
1329
1330 tmpPath.pop_back();
1331 instance.m_Path = tmpPath.Path();
1332 instance.m_PageNumber = path.GetPageNumber();
1333
1334 retval.push_back( instance );
1335 }
1336
1337 return retval;
1338}
1339
1340
1342{
1343 for( const SCH_SHEET_PATH& instance : *this )
1344 {
1345 if( !instance.GetPageNumber().IsEmpty() )
1346 return false;
1347 }
1348
1349 return true;
1350}
1351
1352
1354{
1355 // Don't accidentally renumber existing sheets.
1356 wxCHECK( AllSheetPageNumbersEmpty(), /* void */ );
1357
1358 wxString tmp;
1359 int pageNumber = 1;
1360
1361 for( SCH_SHEET_PATH& instance : *this )
1362 {
1363 tmp.Printf( "%d", pageNumber );
1364 instance.SetPageNumber( tmp );
1365 pageNumber += 1;
1366 }
1367}
1368
1369
1371 const wxString& aProjectName )
1372{
1373 for( SCH_SHEET_PATH& sheetPath : *this )
1374 sheetPath.AddNewSymbolInstances( aPrefixSheetPath, aProjectName );
1375}
1376
1377
1379{
1380 for( SCH_SHEET_PATH& sheetPath : *this )
1381 sheetPath.RemoveSymbolInstances( aPrefixSheetPath );
1382}
1383
1384
1386 int aLastVirtualPageNumber )
1387{
1388 wxString pageNumber;
1389 int lastUsedPageNumber = 1;
1390 int nextVirtualPageNumber = aLastVirtualPageNumber;
1391
1392 // Fetch the list of page numbers already in use.
1393 std::vector< wxString > usedPageNumbers;
1394
1395 if( aPrefixSheetPath.size() )
1396 {
1397 SCH_SHEET_LIST prefixHierarchy( aPrefixSheetPath.at( 0 ) );
1398
1399 for( const SCH_SHEET_PATH& path : prefixHierarchy )
1400 {
1401 pageNumber = path.GetPageNumber();
1402
1403 if( !pageNumber.IsEmpty() )
1404 usedPageNumbers.emplace_back( pageNumber );
1405 }
1406 }
1407
1408 for( SCH_SHEET_PATH& sheetPath : *this )
1409 {
1410 KIID_PATH tmp = sheetPath.Path();
1411 SCH_SHEET_PATH newSheetPath( aPrefixSheetPath );
1412
1413 // Prefix the new hierarchical path.
1414 newSheetPath = newSheetPath + sheetPath;
1415
1416 // Sheets cannot have themselves in the path.
1417 tmp.pop_back();
1418
1419 SCH_SHEET* sheet = sheetPath.Last();
1420
1421 wxCHECK2( sheet, continue );
1422
1423 nextVirtualPageNumber += 1;
1424
1425 SCH_SHEET_INSTANCE instance;
1426
1427 // Add the instance if it doesn't already exist
1428 if( !sheet->getInstance( instance, tmp, true ) )
1429 {
1430 sheet->addInstance( tmp );
1431 sheet->getInstance( instance, tmp, true );
1432 }
1433
1434 // Get a new page number if we don't have one
1435 if( instance.m_PageNumber.IsEmpty() )
1436 {
1437 // Generate the next available page number.
1438 do
1439 {
1440 pageNumber.Printf( wxT( "%d" ), lastUsedPageNumber );
1441 lastUsedPageNumber += 1;
1442 } while( std::find( usedPageNumbers.begin(), usedPageNumbers.end(), pageNumber ) !=
1443 usedPageNumbers.end() );
1444
1445 instance.m_PageNumber = pageNumber;
1446 newSheetPath.SetVirtualPageNumber( nextVirtualPageNumber );
1447 }
1448
1449 newSheetPath.SetPageNumber( instance.m_PageNumber );
1450 usedPageNumbers.push_back( instance.m_PageNumber );
1451 }
1452}
1453
1454
1455void SCH_SHEET_LIST::CheckForMissingSymbolInstances( const wxString& aProjectName )
1456{
1457 for( SCH_SHEET_PATH& sheetPath : *this )
1458 sheetPath.CheckForMissingSymbolInstances( aProjectName );
1459}
1460
1461
1463{
1464 int lastVirtualPageNumber = 1;
1465
1466 for( const SCH_SHEET_PATH& sheetPath : *this )
1467 {
1468 if( sheetPath.GetVirtualPageNumber() > lastVirtualPageNumber )
1469 lastVirtualPageNumber = sheetPath.GetVirtualPageNumber();
1470 }
1471
1472 return lastVirtualPageNumber;
1473}
1474
1475
1476bool SCH_SHEET_LIST::HasPath( const KIID_PATH& aPath ) const
1477{
1478 for( const SCH_SHEET_PATH& path : *this )
1479 {
1480 if( path.Path() == aPath )
1481 return true;
1482 }
1483
1484 return false;
1485}
1486
1487
1488bool SCH_SHEET_LIST::ContainsSheet( const SCH_SHEET* aSheet ) const
1489{
1490 for( const SCH_SHEET_PATH& path : *this )
1491 {
1492 for( size_t i = 0; i < path.size(); i++ )
1493 {
1494 if( path.at( i ) == aSheet )
1495 return true;
1496 }
1497 }
1498
1499 return false;
1500}
1501
1502
1503std::optional<SCH_SHEET_PATH> SCH_SHEET_LIST::GetOrdinalPath( const SCH_SCREEN* aScreen ) const
1504{
1505 // Sheet paths with sheets that do not have a screen object are not valid.
1506 if( !aScreen )
1507 return std::nullopt;
1508
1509 for( const SCH_SHEET_PATH& path: *this )
1510 {
1511 if( path.LastScreen() == aScreen )
1512 return std::optional<SCH_SHEET_PATH>( path );
1513 }
1514
1515 return std::nullopt;
1516}
bool IsContentModified() const
Definition: base_screen.h:60
void SetContentModified(bool aModified=true)
Definition: base_screen.h:59
A singleton item of this class is returned for a weak reference that no longer exists.
wxString GetClass() const override
Return the class name.
void SetPosition(const VECTOR2I &) override
static DELETED_SHEET_ITEM * GetInstance()
double Similarity(const SCH_ITEM &aOther) const override
Return a measure of how likely the other object is to represent the same object.
void Rotate(const VECTOR2I &aCenter, bool aRotateCCW) override
Rotate the item around aCenter 90 degrees in the clockwise direction.
void Move(const VECTOR2I &aMoveVector) override
Move the item by aMoveVector to a new position.
wxString GetItemDescription(UNITS_PROVIDER *aUnitsProvider, bool aFull) const override
Return a user-visible description string of this item.
bool operator==(const SCH_ITEM &aOther) const override
void MirrorVertically(int aCenter) override
Mirror item vertically about aCenter.
void MirrorHorizontally(int aCenter) override
Mirror item horizontally about aCenter.
const KIID m_Uuid
Definition: eda_item.h:498
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition: eda_text.h:98
virtual bool IsVisible() const
Definition: eda_text.h:174
virtual void SetVisible(bool aVisible)
Definition: eda_text.cpp:379
EE_TYPE OfType(KICAD_T aType) const
Definition: sch_rtree.h:241
wxString AsString() const
Definition: kiid.cpp:356
Definition: kiid.h:49
wxString AsString() const
Definition: kiid.cpp:246
Define a library symbol object.
Definition: lib_symbol.h:85
bool IsPower() const override
Definition: lib_symbol.cpp:467
int GetUnitCount() const override
Holds all the data relating to one schematic.
Definition: schematic.h:69
wxString GetFileName() const
Helper to retrieve the filename from the root sheet screen.
Definition: schematic.cpp:300
SCHEMATIC_SETTINGS & Settings() const
Definition: schematic.cpp:306
VECTOR2I GetPosition() const override
Definition: sch_field.cpp:1314
wxString GetShownText(const SCH_SHEET_PATH *aPath, bool aAllowExtraText, int aDepth=0) const
Definition: sch_field.cpp:192
void SetText(const wxString &aText) override
Definition: sch_field.cpp:1069
SCH_FIELD * GetField(FIELD_T aFieldType)
Return a mandatory field in this label.
Definition: sch_label.cpp:1838
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
virtual void RunOnChildren(const std::function< void(SCH_ITEM *)> &aFunction, RECURSE_MODE aMode)
Definition: sch_item.h:566
SCHEMATIC * Schematic() const
Search the item hierarchy to find a SCHEMATIC.
Definition: sch_item.cpp:151
int GetUnit() const
Definition: sch_item.h:236
virtual void SetUnit(int aUnit)
Definition: sch_item.h:235
void AutoplaceFields(SCH_SCREEN *aScreen, AUTOPLACE_ALGO aAlgo) override
Definition: sch_label.cpp:566
std::vector< SCH_FIELD > & GetFields()
Definition: sch_label.h:205
Container to create a flattened list of symbols because in a complex hierarchy, a symbol can be used ...
void Annotate(bool aUseSheetNum, int aSheetIntervalId, int aStartNumber, SCH_MULTI_UNIT_REFERENCE_MAP aLockedUnitMap, const SCH_REFERENCE_LIST &aAdditionalRefs, bool aStartAtCurrent=false)
Set the reference designators in the list that have not been annotated.
size_t GetCount() const
void SplitReferences()
Attempt to split all reference designators into a name (U) and number (1).
void AddItem(const SCH_REFERENCE &aItem)
void UpdateAnnotation()
Update the symbol references for the schematic project (or the current sheet).
A helper to define a symbol's reference designator in a schematic.
wxString GetRef() const
void SetSheetNumber(int aSheetNumber)
EE_RTREE & Items()
Get the full RTree, usually for iterating.
Definition: sch_screen.h:112
const wxString & GetFileName() const
Definition: sch_screen.h:147
bool Remove(SCH_ITEM *aItem, bool aUpdateLibSymbol=true)
Remove aItem from the schematic associated with this screen.
Definition: sch_screen.cpp:325
SCHEMATIC * Schematic() const
Definition: sch_screen.cpp:102
void Update(SCH_ITEM *aItem, bool aUpdateLibSymbol=true)
Update aItem's bounding box in the tree.
Definition: sch_screen.cpp:318
void GetSheets(std::vector< SCH_ITEM * > *aItems) const
Similar to Items().OfType( SCH_SHEET_T ), but return the sheets in a deterministic order (L-R,...
A container for handling SCH_SHEET_PATH objects in a flattened hierarchy.
std::optional< SCH_SHEET_PATH > GetSheetPathByKIIDPath(const KIID_PATH &aPath, bool aIncludeLastSheet=true) const
Finds a SCH_SHEET_PATH that matches the provided KIID_PATH.
SCH_ITEM * GetItem(const KIID &aID, SCH_SHEET_PATH *aPathOut=nullptr) const
Fetch a SCH_ITEM by ID.
void GetMultiUnitSymbols(SCH_MULTI_UNIT_REFERENCE_MAP &aRefList, bool aIncludePowerSymbols=true) const
Add a SCH_REFERENCE_LIST object to aRefList for each same-reference set of multi-unit parts in the li...
std::optional< SCH_SHEET_PATH > GetOrdinalPath(const SCH_SCREEN *aScreen) const
Return the ordinal sheet path of aScreen.
void FillItemMap(std::map< KIID, EDA_ITEM * > &aMap)
Fill an item cache for temporary use when many items need to be fetched.
SCH_SHEET_PATH m_currentSheetPath
void TrimToPageNumbers(const std::vector< wxString > &aPageInclusions)
Truncates the list by removing sheet's with page numbers not in the given list.
void SortByPageNumbers(bool aUpdateVirtualPageNums=true)
Sort the list of sheets by page number.
void AddNewSymbolInstances(const SCH_SHEET_PATH &aPrefixSheetPath, const wxString &aProjectName)
Attempt to add new symbol instances for all symbols in this list of sheet paths prefixed with aPrefix...
bool NameExists(const wxString &aSheetName) const
std::vector< SCH_SHEET_INSTANCE > GetSheetInstances() const
Fetch the instance information for all of the sheets in the hierarchy.
void UpdateSheetInstanceData(const std::vector< SCH_SHEET_INSTANCE > &aSheetInstances)
Update all of the sheet instance information using aSheetInstances.
void SetInitialPageNumbers()
Set initial sheet page numbers.
void RemoveSymbolInstances(const SCH_SHEET_PATH &aPrefixSheetPath)
SCH_SHEET_LIST FindAllSheetsForScreen(const SCH_SCREEN *aScreen) const
Return a SCH_SHEET_LIST with a copy of all the SCH_SHEET_PATH using a particular screen.
bool AllSheetPageNumbersEmpty() const
Check all of the sheet instance for empty page numbers.
void GetSymbols(SCH_REFERENCE_LIST &aReferences, bool aIncludePowerSymbols=true, bool aForceIncludeOrphanSymbols=false) const
Add a SCH_REFERENCE object to aReferences for each symbol in the list of sheets.
bool IsModified() const
Check the entire hierarchy for any modifications.
SCH_SHEET_LIST(SCH_SHEET *aSheet=nullptr)
Construct a flattened list of SCH_SHEET_PATH objects from aSheet.
void AnnotatePowerSymbols()
Silently annotate the not yet annotated power symbols of the entire hierarchy of the sheet path list.
int GetLastVirtualPageNumber() const
void UpdateSymbolInstanceData(const std::vector< SCH_SYMBOL_INSTANCE > &aSymbolInstances)
Update all of the symbol instance information using aSymbolInstances.
void GetSheetsWithinPath(std::vector< SCH_SHEET_PATH > &aSheets, const SCH_SHEET_PATH &aSheetPath) const
Add a SCH_SHEET_PATH object to aSheets for each sheet in the list that are contained within aSheetPat...
bool PageNumberExists(const wxString &aPageNumber) const
void SortByHierarchicalPageNumbers(bool aUpdateVirtualPageNums=true)
This works like SortByPageNumbers, but it sorts the sheets first by their hierarchical depth and then...
void AddNewSheetInstances(const SCH_SHEET_PATH &aPrefixSheetPath, int aLastVirtualPageNumber)
bool ContainsSheet(const SCH_SHEET *aSheet) const
std::vector< KIID_PATH > GetPaths() const
void GetSymbolsWithinPath(SCH_REFERENCE_LIST &aReferences, const SCH_SHEET_PATH &aSheetPath, bool aIncludePowerSymbols=true, bool aForceIncludeOrphanSymbols=false) const
Add a SCH_REFERENCE object to aReferences for each symbol in the list of sheets that are contained wi...
void BuildSheetList(SCH_SHEET *aSheet, bool aCheckIntegrity)
Build the list of sheets and their sheet path from aSheet.
SCH_SHEET_PATH FindSheetForScreen(const SCH_SCREEN *aScreen)
Return the first SCH_SHEET_PATH object (not necessarily the only one) using a particular screen.
void CheckForMissingSymbolInstances(const wxString &aProjectName)
bool HasPath(const KIID_PATH &aPath) const
SCH_SHEET_PATH * FindSheetForPath(const SCH_SHEET_PATH *aPath)
Return a pointer to the first SCH_SHEET_PATH object (not necessarily the only one) matching the provi...
bool TestForRecursion(const SCH_SHEET_LIST &aSrcSheetHierarchy, const wxString &aDestFileName)
Test every SCH_SHEET_PATH in this SCH_SHEET_LIST to verify if adding the sheets stored in aSrcSheetHi...
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
bool GetExcludedFromBOM() const
SCH_ITEM * GetItem(const KIID &aID) const
Fetch a SCH_ITEM by ID.
void AppendMultiUnitSymbol(SCH_MULTI_UNIT_REFERENCE_MAP &aRefList, SCH_SYMBOL *aSymbol, bool aIncludePowerSymbols=true) const
Append a SCH_REFERENCE_LIST object to aRefList based on aSymbol, storing same-reference set of multi-...
const SCH_SHEET * GetSheet(unsigned aIndex) const
bool empty() const
Forwarded method from std::vector.
void GetSymbols(SCH_REFERENCE_LIST &aReferences, bool aIncludePowerSymbols=true, bool aForceIncludeOrphanSymbols=false) const
Adds SCH_REFERENCE object to aReferences for each symbol in the sheet.
int ComparePageNum(const SCH_SHEET_PATH &aSheetPathToTest) const
Compare sheets by their page number.
size_t GetCurrentHash() const
wxString PathHumanReadable(bool aUseShortRootName=true, bool aStripTrailingSeparator=false) const
Return the sheet path in a human readable form made from the sheet names.
bool operator==(const SCH_SHEET_PATH &d1) const
void AddNewSymbolInstances(const SCH_SHEET_PATH &aPrefixSheetPath, const wxString &aProjectName)
Attempt to add new symbol instances for all symbols in this sheet path prefixed with aPrefixSheetPath...
KIID_PATH Path() const
Get the sheet path as an KIID_PATH.
bool TestForRecursion(const wxString &aSrcFileName, const wxString &aDestFileName)
Test the SCH_SHEET_PATH file names to check adding the sheet stored in the file aSrcFileName to the s...
void UpdateAllScreenReferences() const
Update all the symbol references for this sheet path.
bool IsFullPath() const
void MakeFilePathRelativeToParentSheet()
Make the sheet file name relative to its parent sheet.
wxString GetCachedPageNumber() const
std::vector< SCH_SHEET * > m_sheets
SCH_SCREEN * LastScreen()
int Cmp(const SCH_SHEET_PATH &aSheetPathToTest) const
Compare if this is the same sheet path as aSheetPathToTest.
void GetMultiUnitSymbols(SCH_MULTI_UNIT_REFERENCE_MAP &aRefList, bool aIncludePowerSymbols=true) const
Add a SCH_REFERENCE_LIST object to aRefList for each same-reference set of multi-unit parts in the sh...
void initFromOther(const SCH_SHEET_PATH &aOther)
wxString m_cached_page_number
wxString GetPageNumber() const
void RemoveSymbolInstances(const SCH_SHEET_PATH &aPrefixSheetPath)
void CheckForMissingSymbolInstances(const wxString &aProjectName)
bool IsContainedWithin(const SCH_SHEET_PATH &aSheetPathToTest) const
Check if this path is contained inside aSheetPathToTest.
SCH_SHEET * at(size_t aIndex) const
Forwarded method from std::vector.
void SetVirtualPageNumber(int aPageNumber)
Set the sheet instance virtual page number.
void AppendSymbol(SCH_REFERENCE_LIST &aReferences, SCH_SYMBOL *aSymbol, bool aIncludePowerSymbols=true, bool aForceIncludeOrphanSymbols=false) const
Append a SCH_REFERENCE object to aReferences based on aSymbol.
std::map< std::pair< wxString, wxString >, bool > m_recursion_test_cache
bool GetExcludedFromSim() const
wxString PathAsString() const
Return the path of time stamps which do not changes even when editing sheet parameters.
bool GetExcludedFromBoard() const
void SetPageNumber(const wxString &aPageNumber)
Set the sheet instance user definable page number.
SCH_SHEET_PATH & operator=(const SCH_SHEET_PATH &aOther)
bool GetDNP() const
SCH_SHEET * Last() const
Return a pointer to the last SCH_SHEET of the list.
SCH_SHEET_PATH operator+(const SCH_SHEET_PATH &aOther)
int m_virtualPageNumber
Page numbers are maintained by the sheet load order.
void push_back(SCH_SHEET *aSheet)
Forwarded method from std::vector.
size_t size() const
Forwarded method from std::vector.
int GetVirtualPageNumber() const
void pop_back()
Forwarded method from std::vector.
Sheet symbol placed in a schematic, and is the entry point for a sub schematic.
Definition: sch_sheet.h:47
void SetFileName(const wxString &aFilename)
Definition: sch_sheet.h:327
wxString GetFileName() const
Return the filename corresponding to this sheet.
Definition: sch_sheet.h:321
bool IsRootSheet() const
Definition: sch_sheet.cpp:169
bool getInstance(SCH_SHEET_INSTANCE &aInstance, const KIID_PATH &aSheetPath, bool aTestFromEnd=false) const
Definition: sch_sheet.cpp:1395
bool addInstance(const KIID_PATH &aInstance)
Add a new instance aSheetPath to the instance list.
Definition: sch_sheet.cpp:1372
SCH_FIELD * GetField(FIELD_T aFieldType)
Return a mandatory field in this sheet.
Definition: sch_sheet.cpp:369
wxString GetName() const
Definition: sch_sheet.h:113
SCH_SCREEN * GetScreen() const
Definition: sch_sheet.h:116
static int ComparePageNum(const wxString &aPageNumberA, const wxString &aPageNumberB)
Compare page numbers of schematic sheets.
Definition: sch_sheet.cpp:1530
void setPageNumber(const KIID_PATH &aInstance, const wxString &aPageNumber)
Set the page number for the sheet instance aInstance.
Definition: sch_sheet.cpp:1464
wxString getPageNumber(const KIID_PATH &aInstance) const
Return the sheet page number for aInstance.
Definition: sch_sheet.cpp:1447
Schematic symbol object.
Definition: sch_symbol.h:75
void UpdatePrefix()
Set the prefix based on the current reference designator.
Definition: sch_symbol.cpp:703
const std::vector< SCH_SYMBOL_INSTANCE > & GetInstances() const
Definition: sch_symbol.h:134
void RemoveInstance(const SCH_SHEET_PATH &aInstancePath)
Definition: sch_symbol.cpp:483
void SetValueFieldText(const wxString &aValue)
Definition: sch_symbol.cpp:791
void SetFootprintFieldText(const wxString &aFootprint)
Definition: sch_symbol.cpp:807
void AddHierarchicalReference(const KIID_PATH &aPath, const wxString &aRef, int aUnit)
Add a full hierarchical reference to this symbol.
Definition: sch_symbol.cpp:511
bool GetInstance(SCH_SYMBOL_INSTANCE &aInstance, const KIID_PATH &aSheetPath, bool aTestFromEnd=false) const
Definition: sch_symbol.cpp:459
int GetUnitSelection(const SCH_SHEET_PATH *aSheet) const
Return the instance-specific unit selection for the given sheet path.
Definition: sch_symbol.cpp:739
std::unique_ptr< LIB_SYMBOL > & GetLibSymbolRef()
Definition: sch_symbol.h:183
const wxString GetRef(const SCH_SHEET_PATH *aSheet, bool aIncludeUnit=false) const override
Definition: sch_symbol.cpp:611
wxString GetPrefix() const
Definition: sch_symbol.h:249
SCH_FIELD * GetField(FIELD_T aFieldType)
Return a mandatory field in this symbol.
Definition: sch_symbol.cpp:813
static bool empty(const wxTextEntryBase *aCtrl)
#define _(s)
const wxChar *const tracePathsAndFiles
Flag to enable path and file name debug output.
const wxChar *const traceSchSheetPaths
Flag to enable debug output of schematic symbol sheet path manipulation code.
static constexpr void hash_combine(std::size_t &seed)
This is a dummy function to take the final case of hash_combine below.
Definition: hash.h:32
wxString GetRefDesUnannotated(const wxString &aSource)
Return an unannotated refdes from either a prefix or an existing refdes.
STL namespace.
Collection of utility functions for component reference designators (refdes)
@ AUTOPLACE_AUTO
Definition: sch_item.h:70
Definition of the SCH_SHEET_PATH and SCH_SHEET_LIST classes for Eeschema.
std::map< wxString, SCH_REFERENCE_LIST > SCH_MULTI_UNIT_REFERENCE_MAP
Container to map reference designators for multi-unit parts.
A simple container for sheet instance information.
A simple container for schematic symbol instance information.
Definition for symbol library class.
VECTOR2I end
wxLogTrace helper definitions.
@ SCH_SYMBOL_T
Definition: typeinfo.h:172
@ NOT_USED
the 3d code uses this value
Definition: typeinfo.h:79
@ SCH_GLOBAL_LABEL_T
Definition: typeinfo.h:168
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695