KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
dialog_change_symbols.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) 2020-2021 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * @author Wayne Stambaugh <stambaughw@gmail.com>
8 *
9 * This program is free software: you can redistribute it and/or modify it
10 * under the terms of the GNU General Public License as published by the
11 * Free Software Foundation, either version 3 of the License, or (at your
12 * option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful, but
15 * WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License along
20 * with this program. If not, see <http://www.gnu.org/licenses/>.
21 */
22
23#include <algorithm>
24
25#include <bitmaps.h>
26#include <connection_graph.h>
27#include <string_utils.h> // WildCompareString
28#include <kiway.h>
29#include <refdes_utils.h>
30#include <core/kicad_algo.h>
32#include <sch_symbol.h>
33#include <sch_edit_frame.h>
34#include <sch_screen.h>
35#include <schematic.h>
36#include <template_fieldnames.h>
39#include <sch_commit.h>
40
41bool g_selectRefDes = false;
42bool g_selectValue = false;
43 // { change, update }
44bool g_removeExtraFields[2] = { false, false };
45bool g_resetEmptyFields[2] = { false, false };
46bool g_resetFieldText[2] = { true, true };
47bool g_resetFieldVisibilities[2] = { true, false };
48bool g_resetFieldEffects[2] = { true, false };
49bool g_resetFieldPositions[2] = { true, false };
50bool g_resetAttributes[2] = { true, false };
51bool g_resetCustomPower[2] = { false, false };
52
53
55 MODE aMode ) :
57 m_symbol( aSymbol),
58 m_mode( aMode )
59{
60 wxASSERT( aParent );
61
62 if( m_mode == MODE::UPDATE )
63 {
64 m_newIdSizer->Show( false );
65 }
66 else
67 {
68 m_matchAll->SetLabel( _( "Change all symbols in schematic" ) );
69 SetTitle( _( "Change Symbols" ) );
70 m_matchSizer->FindItem( m_matchAll )->Show( false );
71 }
72
73 if( m_symbol )
74 {
75 SCH_SHEET_PATH* currentSheet = &aParent->Schematic().CurrentSheet();
76
77 if( m_mode == MODE::CHANGE )
78 m_matchBySelection->SetLabel( _( "Change selected symbol(s)" ) );
79
80 m_newId->ChangeValue( UnescapeString( m_symbol->GetLibId().Format() ) );
81 m_specifiedReference->ChangeValue( m_symbol->GetRef( currentSheet ) );
82 m_specifiedValue->ChangeValue( UnescapeString( m_symbol->GetField( FIELD_T::VALUE )->GetText() ) );
83 m_specifiedId->ChangeValue( UnescapeString( m_symbol->GetLibId().Format() ) );
84 }
85 else
86 {
87 m_matchSizer->FindItem( m_matchBySelection )->Show( false );
88 }
89
90 m_matchIdBrowserButton->SetBitmap( KiBitmapBundle( BITMAPS::small_library ) );
91 m_newIdBrowserButton->SetBitmap( KiBitmapBundle( BITMAPS::small_library ) );
92
93 if( m_mode == MODE::CHANGE )
94 {
95 m_matchByReference->SetLabel( _( "Change symbols matching reference designator:" ) );
96 m_matchByValue->SetLabel( _( "Change symbols matching value:" ) );
97 m_matchById->SetLabel( _( "Change symbols matching library identifier:" ) );
98 }
99
100 m_matchSizer->SetEmptyCellSize( wxSize( 0, 0 ) );
101 m_matchSizer->Layout();
102
103 for( FIELD_T fieldId : MANDATORY_FIELDS )
104 {
105 int listIdx = m_fieldsBox->GetCount();
106
107 m_fieldsBox->Append( GetDefaultFieldName( fieldId, DO_TRANSLATE ) );
108
109 if( fieldId == FIELD_T::REFERENCE )
110 m_fieldsBox->Check( listIdx, g_selectRefDes );
111 else if( fieldId == FIELD_T::VALUE )
112 m_fieldsBox->Check( listIdx, g_selectValue );
113 else
114 m_fieldsBox->Check( listIdx, true );
115
116 m_mandatoryFieldListIndexes[fieldId] = listIdx;
117 }
118
120 m_messagePanel->SetFileName( Prj().GetProjectPath() + wxT( "report.txt" ) );
121
122 if( aSymbol && aSymbol->IsSelected() )
123 {
124 m_matchBySelection->SetValue( true );
125 }
126 else
127 {
128 if( aMode == MODE::UPDATE )
129 m_matchAll->SetValue( true );
130 else
131 m_matchByReference->SetValue( true );
132 }
133
135
136 if( m_mode == MODE::CHANGE )
137 {
138 m_updateFieldsSizer->GetStaticBox()->SetLabel( _( "Update Fields" ) );
139 m_removeExtraBox->SetLabel( _( "Remove fields if not in new symbol" ) );
140 m_resetEmptyFields->SetLabel( _( "Reset fields if empty in new symbol" ) );
141 m_resetFieldText->SetLabel( _( "Update field text" ) );
142 m_resetFieldVisibilities->SetLabel( _( "Update field visibilities" ) );
143 m_resetFieldEffects->SetLabel( _( "Update field sizes and styles" ) );
144 m_resetFieldPositions->SetLabel( _( "Update field positions" ) );
145 m_resetAttributes->SetLabel( _( "Update symbol attributes" ) );
146 }
147
148 m_removeExtraBox->SetValue( g_removeExtraFields[ (int) m_mode ] );
149 m_resetEmptyFields->SetValue( g_resetEmptyFields[ (int) m_mode ] );
150 m_resetFieldText->SetValue( g_resetFieldText[ (int) m_mode ] );
152 m_resetFieldEffects->SetValue( g_resetFieldEffects[ (int) m_mode ] );
154 m_resetAttributes->SetValue( g_resetAttributes[ (int) m_mode ] );
155 m_resetCustomPower->SetValue( g_resetCustomPower[ (int) m_mode ] );
156
157 // DIALOG_SHIM needs a unique hash_key because classname is not sufficient
158 // because the update and change versions of this dialog have different controls.
159 m_hash_key = TO_UTF8( GetTitle() );
160
161 wxString okLabel = m_mode == MODE::CHANGE ? _( "Change" ) : _( "Update" );
162
163 SetupStandardButtons( { { wxID_OK, okLabel },
164 { wxID_CANCEL, _( "Close" ) } } );
165
166 // Now all widgets have the size fixed, call FinishDialogSettings
168}
169
170
171void DIALOG_CHANGE_SYMBOLS::onMatchByAll( wxCommandEvent& aEvent )
172{
174}
175
176
177void DIALOG_CHANGE_SYMBOLS::onMatchBySelected( wxCommandEvent& aEvent )
178{
180}
181
182
183void DIALOG_CHANGE_SYMBOLS::onMatchByReference( wxCommandEvent& aEvent )
184{
186 m_specifiedReference->SetFocus();
187}
188
189
190void DIALOG_CHANGE_SYMBOLS::onMatchByValue( wxCommandEvent& aEvent )
191{
193 m_specifiedValue->SetFocus();
194}
195
196
197void DIALOG_CHANGE_SYMBOLS::onMatchById( wxCommandEvent& aEvent )
198{
200 m_specifiedId->SetFocus();
201}
202
203
205{
207 event.Skip(); // Mandatory in wxFocusEvent
208}
209
210
212{
214 event.Skip(); // Mandatory in wxFocusEvent
215}
216
217
219{
221 event.Skip(); // Mandatory in wxFocusEvent
222}
223
224
226{
227 g_selectRefDes = m_fieldsBox->IsChecked( m_mandatoryFieldListIndexes[FIELD_T::REFERENCE] );
228 g_selectValue = m_fieldsBox->IsChecked( m_mandatoryFieldListIndexes[FIELD_T::VALUE] );
229
230 g_removeExtraFields[ (int) m_mode ] = m_removeExtraBox->GetValue();
231 g_resetEmptyFields[ (int) m_mode ] = m_resetEmptyFields->GetValue();
232 g_resetFieldText[ (int) m_mode ] = m_resetFieldText->GetValue();
234 g_resetFieldEffects[ (int) m_mode ] = m_resetFieldEffects->GetValue();
236 g_resetAttributes[ (int) m_mode ] = m_resetAttributes->GetValue();
237 g_resetCustomPower[ (int) m_mode ] = m_resetCustomPower->GetValue();
238}
239
240
241wxString getLibIdValue( const wxTextCtrl* aCtrl )
242{
243 wxString rawValue = aCtrl->GetValue();
244 wxString itemName;
245 wxString libName = rawValue.BeforeFirst( ':', &itemName );
246
247 return EscapeString( libName, CTX_LIBID ) + ':' + EscapeString( itemName, CTX_LIBID );
248}
249
250
252{
253 wxString newName = getLibIdValue( m_specifiedId );
254
255 if( KIWAY_PLAYER* frame = Kiway().Player( FRAME_SYMBOL_CHOOSER, true, this ) )
256 {
257 if( frame->ShowModal( &newName, this ) )
258 {
259 m_specifiedId->SetValue( UnescapeString( newName ) );
261 }
262
263 frame->Destroy();
264 }
265}
266
267
269{
270 wxString newName = getLibIdValue( m_newId );
271
272 if( KIWAY_PLAYER* frame = Kiway().Player( FRAME_SYMBOL_CHOOSER, true, this ) )
273 {
274 if( frame->ShowModal( &newName, this ) )
275 {
276 m_newId->SetValue( UnescapeString( newName ) );
278 }
279
280 frame->Destroy();
281 }
282}
283
284
286{
287 SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
288
289 wxCHECK( frame, /* void */ );
290
291 // Load non-mandatory fields from all matching symbols and their library symbols
292 std::set<wxString> fieldNames;
293
294 for( SCH_SHEET_PATH& instance : frame->Schematic().Hierarchy() )
295 {
296 SCH_SCREEN* screen = instance.LastScreen();
297
298 wxCHECK2( screen, continue );
299
300 for( SCH_ITEM* item : screen->Items().OfType( SCH_SYMBOL_T ) )
301 {
302 SCH_SYMBOL* symbol = dynamic_cast<SCH_SYMBOL*>( item );
303
304 wxCHECK2( symbol, continue );
305
306 if( !isMatch( symbol, &instance ) )
307 continue;
308
309 for( SCH_FIELD& field : symbol->GetFields() )
310 {
311 if( !field.IsMandatory() && !field.IsPrivate() )
312 fieldNames.insert( field.GetName() );
313 }
314
315 if( m_mode == MODE::UPDATE && symbol->GetLibId().IsValid() )
316 {
317 LIB_SYMBOL* libSymbol = frame->GetLibSymbol( symbol->GetLibId() );
318
319 if( libSymbol )
320 {
321 std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
322 std::vector<SCH_FIELD*> orderedLibFields;
323
324 flattenedSymbol->GetFields( orderedLibFields );
325
326 for( SCH_FIELD* libField : orderedLibFields )
327 {
328 if( !libField->IsMandatory() && !libField->IsPrivate() )
329 fieldNames.insert( libField->GetName() );
330 }
331 }
332 }
333 }
334 }
335
336 // Load non-mandatory fields from the change-to library symbol
337 if( m_mode == MODE::CHANGE )
338 {
339 LIB_ID newId;
340
341 newId.Parse( getLibIdValue( m_newId ) );
342
343 if( newId.IsValid() )
344 {
345 LIB_SYMBOL* libSymbol = frame->GetLibSymbol( newId );
346
347 if( libSymbol )
348 {
349 std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
350 std::vector<SCH_FIELD*> orderedLibFields;
351
352 flattenedSymbol->GetFields( orderedLibFields );
353
354 for( SCH_FIELD* libField : orderedLibFields )
355 {
356 if( !libField->IsMandatory() && !libField->IsPrivate() )
357 fieldNames.insert( libField->GetName() );
358 }
359 }
360 }
361 }
362
363 // Update the listbox widget
364 wxArrayInt checkedItems;
365 wxArrayString checkedNames;
366
367 m_fieldsBox->GetCheckedItems( checkedItems );
368
369 for( int ii : checkedItems )
370 checkedNames.push_back( m_fieldsBox->GetString( ii ) );
371
372 bool allChecked = true;
373
374 for( int ii = 0; ii < (int) m_fieldsBox->GetCount(); ++ii )
375 {
376 if( !m_fieldsBox->IsChecked( ii )
377 && ii != m_mandatoryFieldListIndexes[FIELD_T::REFERENCE]
378 && ii != m_mandatoryFieldListIndexes[FIELD_T::VALUE] )
379 {
380 allChecked = false;
381 }
382 }
383
384 auto isMandatoryField =
385 [&]( int listbox_idx )
386 {
387 for( FIELD_T fieldId : MANDATORY_FIELDS )
388 {
389 if( m_mandatoryFieldListIndexes[fieldId] == listbox_idx )
390 return true;
391 }
392
393 return false;
394 };
395
396 for( int ii = (int) m_fieldsBox->GetCount() - 1; ii >= 0; --ii )
397 {
398 if( isMandatoryField( ii ) )
399 continue;
400
401 m_fieldsBox->Delete( ii );
402 }
403
404 for( const wxString& fieldName : fieldNames )
405 {
406 m_fieldsBox->Append( fieldName );
407
408 if( allChecked || alg::contains( checkedNames, fieldName ) )
409 m_fieldsBox->Check( m_fieldsBox->GetCount() - 1, true );
410 }
411}
412
413
415{
416 for( unsigned i = 0; i < m_fieldsBox->GetCount(); ++i )
417 m_fieldsBox->Check( i, aCheck );
418}
419
420
421void DIALOG_CHANGE_SYMBOLS::onOkButtonClicked( wxCommandEvent& aEvent )
422{
423 SCH_EDIT_FRAME* parent = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
424
425 wxCHECK( parent, /* void */ );
426
427 wxBusyCursor dummy;
428 SCH_COMMIT commit( parent );
429
431 m_messagePanel->Flush( false );
432
433 // Create the set of fields to be updated. Use non translated (canonical) names
434 // for mandatory fields
435 m_updateFields.clear();
436
437 for( unsigned ii = 0; ii < m_fieldsBox->GetCount(); ++ii )
438 {
439 if( m_fieldsBox->IsChecked( ii ) )
440 m_updateFields.insert( m_fieldsBox->GetString( ii ) );
441 }
442
443 for( FIELD_T fieldId : MANDATORY_FIELDS )
444 {
445 if( m_fieldsBox->IsChecked( m_mandatoryFieldListIndexes[fieldId] ) )
446 m_updateFields.insert( GetCanonicalFieldName( fieldId ) );
447 }
448
449 if( processMatchingSymbols( &commit) )
450 commit.Push( m_mode == MODE::CHANGE ? _( "Change Symbols" ) : _( "Update Symbols" ) );
451
452 m_messagePanel->Flush( false );
453}
454
455
457{
458 SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
459
460 wxCHECK( frame, false );
461
462 if( !aSymbol )
463 return false;
464
465 if( m_matchAll->GetValue() )
466 {
467 return true;
468 }
469 else if( m_matchBySelection->GetValue() )
470 {
471 return aSymbol == m_symbol || aSymbol->IsSelected();
472 }
473 else if( m_matchByReference->GetValue() )
474 {
475 return WildCompareString( m_specifiedReference->GetValue(),
476 UnescapeString( aSymbol->GetRef( aInstance, false ) ),
477 false );
478 }
479 else if( m_matchByValue->GetValue() )
480 {
481 return WildCompareString( m_specifiedValue->GetValue(),
482 UnescapeString( aSymbol->GetField( FIELD_T::VALUE )->GetText() ),
483 false );
484 }
485 else if( m_matchById )
486 {
487 LIB_ID id;
488
490 return aSymbol->GetLibId() == id;
491 }
492
493 return false;
494}
495
496
498{
499 SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
500
501 wxCHECK( frame, false );
502
503 LIB_ID newId;
504 wxString msg;
505 int matchesProcessed = 0;
506 SCH_SYMBOL* symbol = nullptr;
507
508 if( m_mode == MODE::CHANGE )
509 {
510 newId.Parse( getLibIdValue( m_newId ) );
511
512 if( !newId.IsValid() )
513 return false;
514 }
515
516 std::map<SCH_SYMBOL*, SYMBOL_CHANGE_INFO> symbols;
517
518 for( SCH_SHEET_PATH& instance : frame->Schematic().Hierarchy() )
519 {
520 SCH_SCREEN* screen = instance.LastScreen();
521
522 wxCHECK2( screen, continue );
523
524 // Fetch all the symbols that meet the change criteria.
525 for( SCH_ITEM* item : screen->Items().OfType( SCH_SYMBOL_T ) )
526 {
527 symbol = static_cast<SCH_SYMBOL*>( item );
528
529 wxCHECK2( symbol, continue );
530
531 if( !isMatch( symbol, &instance ) )
532 continue;
533
534 if( m_mode == MODE::UPDATE )
535 newId = symbol->GetLibId();
536
537 auto it = symbols.find( symbol );
538
539 if( it == symbols.end() )
540 {
542
543 info.m_Instances.emplace_back( instance );
544 info.m_LibId = newId;
545 symbols.insert( { symbol, info } );
546 }
547 else
548 {
549 it->second.m_Instances.emplace_back( instance );
550 }
551 }
552 }
553
554 if( symbols.size() > 0 )
555 matchesProcessed += processSymbols( aCommit, symbols );
556 else
557 m_messagePanel->Report( _( "*** No symbols matching criteria found ***" ),
559
561
562 return matchesProcessed;
563}
564
565
567 const std::map<SCH_SYMBOL*,
568 SYMBOL_CHANGE_INFO>& aSymbols )
569{
570 wxCHECK( !aSymbols.empty(), 0 );
571
572 int matchesProcessed = 0;
573 SCH_EDIT_FRAME* frame = dynamic_cast<SCH_EDIT_FRAME*>( GetParent() );
574 wxString msg;
575
576 wxCHECK( frame, 0 );
577
578 std::map<SCH_SYMBOL*, SYMBOL_CHANGE_INFO> symbols = aSymbols;
579 std::map<SCH_SYMBOL*, SYMBOL_CHANGE_INFO>::iterator it = symbols.begin();
580
581 // Remove all symbols that don't have a valid library symbol link or enough units to
582 // satisfy the library symbol update.
583 while( it != symbols.end() )
584 {
585 SCH_SYMBOL* symbol = it->first;
586
587 wxCHECK2( symbol && it->second.m_LibId.IsValid(), continue );
588
589 LIB_SYMBOL* libSymbol = frame->GetLibSymbol( it->second.m_LibId );
590
591 if( !libSymbol )
592 {
593 msg = getSymbolReferences( *symbol, it->second.m_LibId );
594 msg << wxT( ": " ) << _( "*** symbol not found ***" );
596 it = symbols.erase( it );
597 continue;
598 }
599
600 std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
601
602 if( flattenedSymbol->GetUnitCount() < symbol->GetUnit() )
603 {
604 msg = getSymbolReferences( *symbol, it->second.m_LibId );
605 msg << wxT( ": " ) << _( "*** new symbol has too few units ***" );
607 it = symbols.erase( it );
608 }
609 else
610 {
611 ++it;
612 }
613 }
614
615 // Removing the symbol needs to be done before the LIB_SYMBOL is changed to prevent stale
616 // library symbols in the schematic file.
617 for( const auto& [ symbol, symbol_change_info ] : symbols )
618 {
619 wxCHECK( symbol && !symbol_change_info.m_Instances.empty(), 0 );
620
621 SCH_SCREEN* screen = symbol_change_info.m_Instances[0].LastScreen();
622
623 wxCHECK( screen, 0 );
624
625 screen->Remove( symbol );
626 SCH_SYMBOL* symbol_copy = static_cast<SCH_SYMBOL*>( symbol->Clone() );
627 aCommit->Modified( symbol, symbol_copy, screen );
628
629 CONNECTION_GRAPH* connectionGraph = screen->Schematic()->ConnectionGraph();
630
631 // When we replace the lib symbol below, we free the associated pins if the new symbol has
632 // fewer than the original. This will cause the connection graph to be out of date unless
633 // we replace references in the graph to the old symbol/pins with references to the ones
634 // stored in the undo stack.
635 if( connectionGraph )
636 connectionGraph->ExchangeItem( symbol, symbol_copy );
637 }
638
639 for( const auto& [ symbol, symbol_change_info ] : symbols )
640 {
641 // Remember initial link before changing for diags purpose
642 wxString initialLibLinkName = UnescapeString( symbol->GetLibId().Format() );
643
644 if( symbol_change_info.m_LibId != symbol->GetLibId() )
645 symbol->SetLibId( symbol_change_info.m_LibId );
646
647 LIB_SYMBOL* libSymbol = frame->GetLibSymbol( symbol_change_info.m_LibId );
648 std::unique_ptr<LIB_SYMBOL> flattenedSymbol = libSymbol->Flatten();
649 SCH_SCREEN* screen = symbol_change_info.m_Instances[0].LastScreen();
650
651 symbol->SetLibSymbol( flattenedSymbol.release() );
652
653 if( m_resetAttributes->GetValue() )
654 {
655 // Fetch the attributes from the *flattened* library symbol. They are not supported
656 // in derived symbols.
657 symbol->SetExcludedFromSim( symbol->GetLibSymbolRef()->GetExcludedFromSim() );
658 symbol->SetExcludedFromBOM( symbol->GetLibSymbolRef()->GetExcludedFromBOM() );
659 symbol->SetExcludedFromBoard( symbol->GetLibSymbolRef()->GetExcludedFromBoard() );
660 }
661
662 if( m_resetPinTextVisibility->GetValue() )
663 {
664 symbol->SetShowPinNames( symbol->GetLibSymbolRef()->GetShowPinNames() );
665 symbol->SetShowPinNumbers( symbol->GetLibSymbolRef()->GetShowPinNumbers() );
666 }
667
668 bool removeExtras = m_removeExtraBox->GetValue();
669 bool resetVis = m_resetFieldVisibilities->GetValue();
670 bool resetEffects = m_resetFieldEffects->GetValue();
671 bool resetPositions = m_resetFieldPositions->GetValue();
672
673 for( int ii = (int) symbol->GetFields().size() - 1; ii >= 0; ii-- )
674 {
675 SCH_FIELD& field = symbol->GetFields()[ii];
676 SCH_FIELD* libField = nullptr;
677 bool doUpdate = field.IsPrivate();
678
679 // Mandatory fields always exist in m_updateFields, but these names can be translated.
680 // so use GetCanonicalName().
681 doUpdate |= alg::contains( m_updateFields, field.GetCanonicalName() );
682
683 if( !doUpdate )
684 continue;
685
686 if( field.IsMandatory() )
687 libField = symbol->GetLibSymbolRef()->GetField( field.GetId() );
688 else
689 libField = symbol->GetLibSymbolRef()->GetField( field.GetName() );
690
691 if( libField )
692 {
693 field.SetPrivate( libField->IsPrivate() );
694
695 bool resetText = libField->GetText().IsEmpty() ? m_resetEmptyFields->GetValue()
696 : m_resetFieldText->GetValue();
697
698 if( resetText )
699 {
700 if( field.GetId() == FIELD_T::REFERENCE )
701 {
702 wxString prefix = UTIL::GetRefDesPrefix( libField->GetText() );
703
704 for( const SCH_SHEET_PATH& instance : symbol_change_info.m_Instances )
705 {
706 wxString ref = symbol->GetRef( &instance );
707 int number = UTIL::GetRefDesNumber( ref );
708
709 if( number >= 0 )
710 ref.Printf( wxS( "%s%d" ), prefix, number );
711 else
712 ref = UTIL::GetRefDesUnannotated( prefix );
713
714 symbol->SetRef( &instance, ref );
715 }
716 }
717 else if( field.GetId() == FIELD_T::VALUE )
718 {
719 if( !symbol->IsPower() || m_resetCustomPower->IsChecked() )
720 symbol->SetValueFieldText( UnescapeString( libField->GetText() ) );
721 }
722 else
723 {
724 field.SetText( libField->GetText() );
725 }
726 }
727
728 if( resetVis )
729 field.SetVisible( libField->IsVisible() );
730
731 if( resetEffects )
732 {
733 // Careful: the visible bit and position are also set by SetAttributes()
734 bool visible = field.IsVisible();
735 VECTOR2I pos = field.GetPosition();
736
737 field.SetAttributes( *libField );
738
739 field.SetVisible( visible );
740 field.SetPosition( pos );
741 field.SetNameShown( libField->IsNameShown() );
742 field.SetCanAutoplace( libField->CanAutoplace() );
743 }
744
745 if( resetPositions )
746 field.SetTextPos( symbol->GetPosition() + libField->GetTextPos() );
747 }
748 else if( !field.IsMandatory() && removeExtras )
749 {
750 symbol->RemoveField( field.GetName() );
751 }
752 }
753
754 std::vector<SCH_FIELD*> libFields;
755 symbol->GetLibSymbolRef()->GetFields( libFields );
756
757 for( SCH_FIELD* libField : libFields )
758 {
759 if( libField->IsMandatory() )
760 continue;
761
762 if( !alg::contains( m_updateFields, libField->GetCanonicalName() ) )
763 continue;
764
765 if( !symbol->GetField( libField->GetName() ) )
766 {
767 SCH_FIELD* schField = symbol->AddField( SCH_FIELD( { 0, 0 }, FIELD_T::USER, symbol,
768 libField->GetName() ) );
769
770 // Careful: the visible bit and position are also set by SetAttributes()
771 schField->SetAttributes( *libField );
772 schField->SetText( libField->GetText() );
773 schField->SetTextPos( symbol->GetPosition() + libField->GetTextPos() );
774 schField->SetPrivate( libField->IsPrivate() );
775 }
776
777 if( resetPositions && frame->eeconfig()->m_AutoplaceFields.enable )
778 {
779 AUTOPLACE_ALGO fieldsAutoplaced = symbol->GetFieldsAutoplaced();
780
781 if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
782 symbol->AutoplaceFields( screen, fieldsAutoplaced );
783 }
784 }
785
786 symbol->SetSchSymbolLibraryName( wxEmptyString );
787 screen->Append( symbol );
788
789 if( resetPositions )
790 {
791 AUTOPLACE_ALGO fieldsAutoplaced = symbol->GetFieldsAutoplaced();
792
793 if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
794 symbol->AutoplaceFields( screen, fieldsAutoplaced );
795 }
796
797 frame->GetCanvas()->GetView()->Update( symbol );
798
799 msg = getSymbolReferences( *symbol, symbol_change_info.m_LibId, &initialLibLinkName );
800 msg += wxS( ": OK" );
802 matchesProcessed +=1;
803 }
804
805 return matchesProcessed;
806}
807
808
810 const LIB_ID& aNewId,
811 const wxString* aOldLibLinkName )
812{
813 wxString msg;
814 wxString references;
815 LIB_ID oldId = aSymbol.GetLibId();
816
817 wxString oldLibLinkName; // For report
818
819 if( aOldLibLinkName )
820 oldLibLinkName = *aOldLibLinkName;
821 else
822 oldLibLinkName = UnescapeString( oldId.Format() );
823
824 SCH_EDIT_FRAME* parent = dynamic_cast< SCH_EDIT_FRAME* >( GetParent() );
825
826 wxCHECK( parent, msg );
827
828 SCH_SHEET_LIST sheets = parent->Schematic().Hierarchy();
829
830 for( const SCH_SYMBOL_INSTANCE& instance : aSymbol.GetInstances() )
831 {
832 // Only include the symbol instances for the current project.
833 if( !sheets.HasPath( instance.m_Path ) )
834 continue;
835
836 if( references.IsEmpty() )
837 references = instance.m_Reference;
838 else
839 references += wxT( " " ) + instance.m_Reference;
840 }
841
842 if( m_mode == MODE::UPDATE )
843 {
844 if( aSymbol.GetInstances().size() == 1 )
845 {
846 msg.Printf( _( "Update symbol %s from '%s' to '%s'" ),
847 references,
848 oldLibLinkName,
849 UnescapeString( aNewId.Format() ) );
850 }
851 else
852 {
853 msg.Printf( _( "Update symbols %s from '%s' to '%s'" ),
854 references,
855 oldLibLinkName,
856 UnescapeString( aNewId.Format() ) );
857 }
858 }
859 else // mode is MODE::CHANGE
860 {
861 if( aSymbol.GetInstances().size() == 1 )
862 {
863 msg.Printf( _( "Change symbol %s from '%s' to '%s'" ),
864 references,
865 oldLibLinkName,
866 UnescapeString( aNewId.Format() ) );
867 }
868 else
869 {
870 msg.Printf( _( "Change symbols %s from '%s' to '%s'" ),
871 references,
872 oldLibLinkName,
873 UnescapeString( aNewId.Format() ) );
874 }
875 }
876
877 return msg;
878}
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition: bitmap.cpp:110
COMMIT & Modified(EDA_ITEM *aItem, EDA_ITEM *aCopy, BASE_SCREEN *aScreen=nullptr)
Create an undo entry for an item that has been already modified.
Definition: commit.h:118
Calculate the connectivity of a schematic and generates netlists.
void ExchangeItem(SCH_ITEM *aOldItem, SCH_ITEM *aNewItem)
Replace all references to #aOldItem with #aNewItem in the graph.
Class DIALOG_CHANGE_SYMBOLS_BASE.
WX_HTML_REPORT_PANEL * m_messagePanel
STD_BITMAP_BUTTON * m_matchIdBrowserButton
void onNewLibIDKillFocus(wxFocusEvent &event) override
void onMatchBySelected(wxCommandEvent &aEvent) override
bool isMatch(SCH_SYMBOL *aSymbol, SCH_SHEET_PATH *aInstance)
std::map< FIELD_T, int > m_mandatoryFieldListIndexes
Index in the list control for each mandatory FIELD_T type.
void onOkButtonClicked(wxCommandEvent &aEvent) override
std::set< wxString > m_updateFields
Set of field names that should have values updated.
void onMatchById(wxCommandEvent &aEvent) override
void launchMatchIdSymbolBrowser(wxCommandEvent &aEvent) override
void onMatchByReference(wxCommandEvent &aEvent) override
void onMatchByValue(wxCommandEvent &aEvent) override
int processMatchingSymbols(SCH_COMMIT *aCommit)
DIALOG_CHANGE_SYMBOLS(SCH_EDIT_FRAME *aParent, SCH_SYMBOL *aSymbol, MODE aMode=MODE::UPDATE)
void checkAll(bool aCheck)
Select or deselect all fields in the listbox widget.
void onMatchTextKillFocus(wxFocusEvent &event) override
void onMatchIDKillFocus(wxFocusEvent &event) override
wxString getSymbolReferences(SCH_SYMBOL &aSymbol, const LIB_ID &aNewId, const wxString *aOldLibLinkName=nullptr)
void launchNewIdSymbolBrowser(wxCommandEvent &aEvent) override
void onMatchByAll(wxCommandEvent &aEvent) override
int processSymbols(SCH_COMMIT *aCommit, const std::map< SCH_SYMBOL *, SYMBOL_CHANGE_INFO > &aSymbols)
void SetupStandardButtons(std::map< int, wxString > aLabels={})
std::string m_hash_key
Definition: dialog_shim.h:194
void finishDialogSettings()
In all dialogs, we must call the same functions to fix minimal dlg size, the default position and per...
bool IsSelected() const
Definition: eda_item.h:120
const VECTOR2I & GetTextPos() const
Definition: eda_text.h:260
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
void SetTextPos(const VECTOR2I &aPoint)
Definition: eda_text.cpp:571
void SetAttributes(const EDA_TEXT &aSrc, bool aSetPosition=true)
Set the text attributes from another instance.
Definition: eda_text.cpp:426
virtual void SetVisible(bool aVisible)
Definition: eda_text.cpp:379
AUTOPLACE_FIELDS m_AutoplaceFields
EE_TYPE OfType(KICAD_T aType) const
Definition: sch_rtree.h:241
virtual void Update(const VIEW_ITEM *aItem, int aUpdateFlags) const
For dynamic VIEWs, inform the associated VIEW that the graphical representation of this item has chan...
Definition: view.cpp:1673
PROJECT & Prj() const
Return a reference to the PROJECT associated with this KIWAY.
KIWAY & Kiway() const
Return a reference to the KIWAY that this object has an opportunity to participate in.
Definition: kiway_holder.h:55
A wxFrame capable of the OpenProjectFiles function, meaning it can load a portion of a KiCad project.
Definition: kiway_player.h:65
A logical library item identifier and consists of various portions much like a URI.
Definition: lib_id.h:49
int Parse(const UTF8 &aId, bool aFix=false)
Parse LIB_ID with the information from aId.
Definition: lib_id.cpp:52
bool IsValid() const
Check if this LID_ID is valid.
Definition: lib_id.h:172
UTF8 Format() const
Definition: lib_id.cpp:119
Define a library symbol object.
Definition: lib_symbol.h:85
std::unique_ptr< LIB_SYMBOL > Flatten() const
Return a flattened symbol inheritance to the caller.
Definition: lib_symbol.cpp:342
SCH_SHEET_LIST Hierarchy() const
Return the full schematic flattened hierarchical sheet list.
Definition: schematic.cpp:208
CONNECTION_GRAPH * ConnectionGraph() const
Definition: schematic.h:158
SCH_SHEET_PATH & CurrentSheet() const
Definition: schematic.h:148
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
EESCHEMA_SETTINGS * eeconfig() const
LIB_SYMBOL * GetLibSymbol(const LIB_ID &aLibId, bool aUseCacheLib=false, bool aShowErrorMsg=false)
Load symbol from symbol library table.
virtual void Push(const wxString &aMessage=wxT("A commit"), int aCommitFlags=0) override
Execute the changes.
Definition: sch_commit.cpp:435
KIGFX::SCH_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
Schematic editor (Eeschema) main window.
SCH_SHEET_PATH & GetCurrentSheet() const
SCHEMATIC & Schematic() const
bool IsMandatory() const
Definition: sch_field.cpp:1336
VECTOR2I GetPosition() const override
Definition: sch_field.cpp:1314
bool IsNameShown() const
Definition: sch_field.h:208
FIELD_T GetId() const
Definition: sch_field.h:124
void SetCanAutoplace(bool aCanPlace)
Definition: sch_field.h:220
wxString GetCanonicalName() const
Get a non-language-specific name for a field which can be used for storage, variable look-up,...
Definition: sch_field.cpp:1098
wxString GetName(bool aUseDefaultName=true) const
Return the field name (not translated).
Definition: sch_field.cpp:1084
void SetPosition(const VECTOR2I &aPosition) override
Definition: sch_field.cpp:1294
bool CanAutoplace() const
Definition: sch_field.h:219
void SetText(const wxString &aText) override
Definition: sch_field.cpp:1069
void SetNameShown(bool aShown=true)
Definition: sch_field.h:209
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
void SetPrivate(bool aPrivate)
Definition: sch_item.h:241
int GetUnit() const
Definition: sch_item.h:236
bool IsPrivate() const
Definition: sch_item.h:242
void Append(SCH_ITEM *aItem, bool aUpdateLibSymbol=true)
Definition: sch_screen.cpp:155
EE_RTREE & Items()
Get the full RTree, usually for iterating.
Definition: sch_screen.h:112
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
A container for handling SCH_SHEET_PATH objects in a flattened hierarchy.
bool HasPath(const KIID_PATH &aPath) const
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
void UpdateAllScreenReferences() const
Update all the symbol references for this sheet path.
Schematic symbol object.
Definition: sch_symbol.h:75
const std::vector< SCH_SYMBOL_INSTANCE > & GetInstances() const
Definition: sch_symbol.h:134
void GetFields(std::vector< SCH_FIELD * > &aVector, bool aVisibleOnly) const override
Populate a std::vector with SCH_FIELDs, sorted in ordinal order.
Definition: sch_symbol.cpp:841
const LIB_ID & GetLibId() const override
Definition: sch_symbol.h:164
const wxString GetRef(const SCH_SHEET_PATH *aSheet, bool aIncludeUnit=false) const override
Definition: sch_symbol.cpp:611
SCH_FIELD * GetField(FIELD_T aFieldType)
Return a mandatory field in this symbol.
Definition: sch_symbol.cpp:813
void SetBitmap(const wxBitmapBundle &aBmp)
void Clear()
Clears the report panel.
void SetLazyUpdate(bool aLazyUpdate)
Set the lazy update.
void SetFileName(const wxString &aReportFileName)
Set the report full file name to the string.
void Report(const wxString &aText, SEVERITY aSeverity, REPORTER::LOCATION aLocation=REPORTER::LOC_BODY)
Report the string.
void Flush(bool aSort=false)
Force updating the HTML page, after the report is built in lazy mode If aSort = true,...
bool g_resetFieldEffects[2]
bool g_selectRefDes
bool g_resetFieldPositions[2]
bool g_resetAttributes[2]
bool g_removeExtraFields[2]
bool g_resetFieldVisibilities[2]
bool g_resetEmptyFields[2]
wxString getLibIdValue(const wxTextCtrl *aCtrl)
bool g_resetFieldText[2]
bool g_selectValue
bool g_resetCustomPower[2]
#define _(s)
@ FRAME_SYMBOL_CHOOSER
Definition: frame_type.h:37
wxString GetRefDesPrefix(const wxString &aRefDes)
Get the (non-numeric) prefix from a refdes - e.g.
wxString GetRefDesUnannotated(const wxString &aSource)
Return an unannotated refdes from either a prefix or an existing refdes.
int GetRefDesNumber(const wxString &aRefDes)
Get the numeric suffix from a refdes - e.g.
bool contains(const _Container &__container, _Value __value)
Returns true if the container contains the given value.
Definition: kicad_algo.h:100
Collection of utility functions for component reference designators (refdes)
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_ACTION
AUTOPLACE_ALGO
Definition: sch_item.h:68
@ AUTOPLACE_MANUAL
Definition: sch_item.h:71
@ AUTOPLACE_AUTO
Definition: sch_item.h:70
std::vector< FAB_LAYER_COLOR > dummy
bool WildCompareString(const wxString &pattern, const wxString &string_to_tst, bool case_sensitive)
Compare a string against wild card (* and ?) pattern using the usual rules.
wxString UnescapeString(const wxString &aSource)
wxString EscapeString(const wxString &aSource, ESCAPE_CONTEXT aContext)
The Escape/Unescape routines use HTML-entity-reference-style encoding to handle characters which are:...
#define TO_UTF8(wxstring)
Convert a wxString to a UTF8 encoded C string for all wxWidgets build modes.
Definition: string_utils.h:403
@ CTX_LIBID
Definition: string_utils.h:54
A simple container for schematic symbol instance information.
wxString GetDefaultFieldName(FIELD_T aFieldId, bool aTranslateForHI)
Return a default symbol field name for a mandatory field type.
#define DO_TRANSLATE
#define MANDATORY_FIELDS
FIELD_T
The set of all field indices assuming an array like sequence that a SCH_COMPONENT or LIB_PART can hol...
wxString GetCanonicalFieldName(FIELD_T aFieldType)
@ SCH_SYMBOL_T
Definition: typeinfo.h:172