KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
dialog_lib_symbol_properties.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 The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
25
26#include <pgm_base.h>
27#include <eeschema_settings.h>
28#include <bitmaps.h>
29#include <confirm.h>
31#include <kiway.h>
32#include <symbol_edit_frame.h>
34#include <math/util.h> // for KiROUND
35#include <sch_symbol.h>
36#include <kiplatform/ui.h>
38#include <widgets/wx_grid.h>
40#include <string_utils.h>
41#include <project_sch.h>
42#include <refdes_utils.h>
43#include <dialog_sim_model.h>
44
49
50#include <wx/clipbrd.h>
51#include <wx/msgdlg.h>
52
53
57
58
60 LIB_SYMBOL* aLibEntry ) :
62 m_Parent( aParent ),
63 m_libEntry( aLibEntry ),
64 m_pinNameOffset( aParent, m_nameOffsetLabel, m_nameOffsetCtrl, m_nameOffsetUnits, true ),
65 m_delayedFocusCtrl( nullptr ),
66 m_delayedFocusGrid( nullptr ),
67 m_delayedFocusRow( -1 ),
68 m_delayedFocusColumn( -1 ),
69 m_delayedFocusPage( -1 ),
70 m_fpFilterTricks( std::make_unique<LISTBOX_TRICKS>( *this, *m_FootprintFilterListBox ) )
71{
73 m_NoteBook->AddPage( m_embeddedFiles, _( "Embedded Files" ) );
74
75 // Give a bit more room for combobox editors
76 m_grid->SetDefaultRowSize( m_grid->GetDefaultRowSize() + 4 );
77 m_fields = new FIELDS_GRID_TABLE( this, aParent, m_grid, m_libEntry,
80 m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this, aLibEntry,
81 [&]( wxCommandEvent& aEvent )
82 {
83 OnAddField( aEvent );
84 } ) );
85 m_grid->SetSelectionMode( wxGrid::wxGridSelectRows );
86
87 // Load the FIELDS_GRID_TABLE
89
90 // Show/hide columns according to the user's preference
93
94 m_SymbolNameCtrl->SetValidator( FIELD_VALIDATOR( FIELD_T::VALUE ) );
95
96 // Configure button logos
97 m_bpAdd->SetBitmap( KiBitmapBundle( BITMAPS::small_plus ) );
98 m_bpDelete->SetBitmap( KiBitmapBundle( BITMAPS::small_trash ) );
99 m_bpMoveUp->SetBitmap( KiBitmapBundle( BITMAPS::small_up ) );
100 m_bpMoveDown->SetBitmap( KiBitmapBundle( BITMAPS::small_down ) );
101 m_addFilterButton->SetBitmap( KiBitmapBundle( BITMAPS::small_plus ) );
102 m_deleteFilterButton->SetBitmap( KiBitmapBundle( BITMAPS::small_trash ) );
103 m_editFilterButton->SetBitmap( KiBitmapBundle( BITMAPS::small_edit ) );
104
105 m_btnCreateJumperPinGroup->SetBitmap( KiBitmapBundle( BITMAPS::right ) );
106 m_btnRemoveJumperPinGroup->SetBitmap( KiBitmapBundle( BITMAPS::left ) );
107
109
110 if( aParent->IsSymbolFromLegacyLibrary() && !aParent->IsSymbolFromSchematic() )
111 {
112 m_stdSizerButtonCancel->SetDefault();
113 m_stdSizerButtonOK->SetLabel( _( "Read Only" ) );
114 m_stdSizerButtonOK->Enable( false );
115 }
116
117 // wxFormBuilder doesn't include this event...
118 m_grid->Connect( wxEVT_GRID_CELL_CHANGING,
120 nullptr, this );
121
122 // Forward the delete button to the tricks
123 m_deleteFilterButton->Bind( wxEVT_BUTTON,
124 [&]( wxCommandEvent& aEvent )
125 {
126 wxCommandEvent cmdEvent( EDA_EVT_LISTBOX_DELETE );
127 m_fpFilterTricks->ProcessEvent( cmdEvent );
128 } );
129
130 // When the filter tricks modifies something, update ourselves
131 m_FootprintFilterListBox->Bind( EDA_EVT_LISTBOX_CHANGED,
132 [&]( wxCommandEvent& aEvent )
133 {
134 OnModify();
135 } );
136
138 {
140 && aLibEntry->IsRoot() )
142 && aLibEntry->IsDerived() ) )
143 {
144 resetSize();
145 }
146 }
147
150
151 m_grid->GetParent()->Layout();
153 Layout();
154
156}
157
158
160{
161 m_lastOpenedPage = m_NoteBook->GetSelection( );
162
164 cfg->m_EditSymbolVisibleColumns = m_grid->GetShownColumnsAsString();
165
166 // Prevents crash bug in wxGrid's d'tor
168
169 m_grid->Disconnect( wxEVT_GRID_CELL_CHANGING,
171 nullptr, this );
172
173 // Delete the GRID_TRICKS.
174 m_grid->PopEventHandler( true );
175}
176
177
179{
180 if( !wxDialog::TransferDataToWindow() )
181 return false;
182
183 std::set<wxString> defined;
184
185 for( SCH_FIELD& field : *m_fields )
186 defined.insert( field.GetName() );
187
188 // Add in any template fieldnames not yet defined:
189 // Read global fieldname templates
191 EESCHEMA_SETTINGS* cfg = mgr.GetAppSettings<EESCHEMA_SETTINGS>( "eeschema" );
192
193 if( cfg )
194 {
195 TEMPLATES templateMgr;
196
197 if( !cfg->m_Drawing.field_names.IsEmpty() )
198 templateMgr.AddTemplateFieldNames( cfg->m_Drawing.field_names );
199
200 for( const TEMPLATE_FIELDNAME& templateFieldname : templateMgr.GetTemplateFieldNames() )
201 {
202 if( defined.count( templateFieldname.m_Name ) <= 0 )
203 {
204 SCH_FIELD field( { 0, 0 }, FIELD_T::USER, m_libEntry, templateFieldname.m_Name );
205 field.SetVisible( templateFieldname.m_Visible );
206 m_fields->push_back( field );
207 m_addedTemplateFields.insert( templateFieldname.m_Name );
208 }
209 }
210 }
211
212 // The Y axis for components in lib is from bottom to top while the screen axis is top
213 // to bottom: we must change the y coord sign for editing
214 for( size_t i = 0; i < m_fields->size(); ++i )
215 {
216 VECTOR2I pos = m_fields->at( i ).GetPosition();
217 pos.y = -pos.y;
218 m_fields->at( i ).SetPosition( pos );
219 }
220
221 // notify the grid
222 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, m_fields->GetNumberRows() );
223 m_grid->ProcessTableMessage( msg );
225
227
228 m_KeywordCtrl->ChangeValue( m_libEntry->GetKeyWords() );
231 m_libEntry->GetUnitCount() == 1 );
232
233 // If a symbol contains no body-style-specific pins or graphic items,
234 // symbol->HasAlternateBodyStyle() will return false.
235 // But when editing a symbol with DeMorgan option set, we don't want to keep turning it off
236 // just because there aren't any body-style-specific items yet, so we force it to on if the
237 // parent frame has it enabled.
239
240 m_OptionPower->SetValue( m_libEntry->IsPower() );
242
243 if( m_libEntry->IsPower() )
244 {
245 m_spiceFieldsButton->Hide();
246 m_OptionLocalPower->Enable();
247 }
248 else
249 {
250 m_OptionLocalPower->Enable( false );
251 }
252
256
261
262 wxArrayString tmp = m_libEntry->GetFPFilters();
263 m_FootprintFilterListBox->Append( tmp );
264
266 m_btnCreateJumperPinGroup->Disable();
267 m_btnRemoveJumperPinGroup->Disable();
268
269 std::set<wxString> availablePins;
270
271 for( const SCH_PIN* pin : m_libEntry->GetPins() )
272 availablePins.insert( pin->GetNumber() );
273
274 for( const std::set<wxString>& group : m_libEntry->JumperPinGroups() )
275 {
276 wxString groupTxt;
277 size_t i = 0;
278
279 for( const wxString& pinNumber : group )
280 {
281 availablePins.erase( pinNumber );
282 groupTxt << pinNumber;
283
284 if( ++i < group.size() )
285 groupTxt << ", ";
286 }
287
288 m_listJumperPinGroups->Append( groupTxt );
289 }
290
291 for( const wxString& pin : availablePins )
292 m_listAvailablePins->AppendString( pin );
293
294 // Populate the list of root parts for inherited objects.
295 if( m_libEntry->IsDerived() )
296 {
297 wxArrayString symbolNames;
298 wxString libName = m_Parent->GetCurLib();
299
300 // Someone forgot to set the current library in the editor frame window.
301 wxCHECK( !libName.empty(), false );
302
303 m_Parent->GetLibManager().GetSymbolNames( libName, symbolNames );
304
305 // Sort the list of symbols for easier search
306 symbolNames.Sort(
307 []( const wxString& a, const wxString& b ) -> int
308 {
309 return StrNumCmp( a, b, true );
310 } );
311
312 // Do allow an inherited symbol to be derived from itself.
313 symbolNames.Remove( m_libEntry->GetName() );
314 m_inheritanceSelectCombo->Append( symbolNames );
315
316 if( LIB_SYMBOL_SPTR rootSymbol = m_libEntry->GetParent().lock() )
317 {
318 wxString parentName = UnescapeString( rootSymbol->GetName() );
319 int selection = m_inheritanceSelectCombo->FindString( parentName );
320
321 if( selection == wxNOT_FOUND )
322 return false;
323
324 m_inheritanceSelectCombo->SetSelection( selection );
325 }
326
328 }
329
330 m_NoteBook->SetSelection( (unsigned) m_lastOpenedPage );
331
333
334 return true;
335}
336
337
339{
341 return false;
342
343 // Symbol reference can be empty because it inherits from the parent symbol.
344 if( m_libEntry->IsRoot() )
345 {
346 SCH_FIELD* field = m_fields->GetField( FIELD_T::REFERENCE );
347
348 if( UTIL::GetRefDesPrefix( field->GetText() ).IsEmpty() )
349 {
350 if( m_NoteBook->GetSelection() != 0 )
351 m_NoteBook->SetSelection( 0 );
352
353 m_delayedErrorMessage = _( "References must start with a letter." );
356 m_delayedFocusRow = m_fields->GetFieldRow( FIELD_T::REFERENCE );
358
359 return false;
360 }
361 }
362
363 // Check for missing field names.
364 for( int ii = 0; ii < (int) m_fields->size(); ++ii )
365 {
366 SCH_FIELD& field = m_fields->at( ii );
367
368 if( field.IsMandatory() )
369 continue;
370
371 wxString fieldName = field.GetName( false );
372
373 if( fieldName.IsEmpty() && !field.GetText().IsEmpty() )
374 {
375 if( m_NoteBook->GetSelection() != 0 )
376 m_NoteBook->SetSelection( 0 );
377
378 m_delayedErrorMessage = _( "Fields must have a name." );
383
384 return false;
385 }
386 }
387
388 // Verify that the parent name is set if the symbol is inherited
389 if( m_libEntry->IsDerived() )
390 {
391 wxString parentName = m_inheritanceSelectCombo->GetValue();
392
393 if( parentName.IsEmpty() )
394 {
395 m_delayedErrorMessage = _( "Derived symbol must have a parent selected" );
396
397 return false;
398 }
399 }
400
401 /*
402 * Confirm destructive actions.
403 */
404
405 if( m_SelNumberOfUnits->GetValue() < m_libEntry->GetUnitCount() )
406 {
407 if( !IsOK( this, _( "Delete extra units from symbol?" ) ) )
408 return false;
409 }
410
412 {
413 if( !IsOK( this, _( "Delete alternate body style (De Morgan) from symbol?" ) ) )
414 return false;
415 }
416
417 return true;
418}
419
420
422{
423 if( !wxDialog::TransferDataFromWindow() )
424 return false;
425
427 return false;
428
429 wxString newName = EscapeString( m_SymbolNameCtrl->GetValue(), CTX_LIBID );
430 wxString oldName = m_libEntry->GetName();
431
432 if( newName.IsEmpty() )
433 {
434 wxMessageBox( _( "Symbol must have a name." ) );
435 return false;
436 }
437
438 UNDO_REDO opType = UNDO_REDO::LIBEDIT;
439
440 if( oldName != newName )
441 {
442 wxString libName = m_Parent->GetCurLib();
443
444 if( m_Parent->GetLibManager().SymbolNameInUse( newName, libName ) )
445 {
446 wxString msg;
447
448 msg.Printf( _( "Symbol name '%s' already in use in library '%s'." ),
449 UnescapeString( newName ),
450 libName );
451 DisplayErrorMessage( this, msg );
452 return false;
453 }
454
455 opType = UNDO_REDO::LIB_RENAME;
456 }
457
458 m_Parent->SaveCopyInUndoList( _( "Edit Symbol Properties" ), m_libEntry, opType );
459
460 // The Y axis for components in lib is from bottom to top while the screen axis is top
461 // to bottom: we must change the y coord sign when writing back to the library
462 for( SCH_FIELD& field : *m_fields )
463 {
464 VECTOR2I pos = field.GetPosition();
465 pos.y = -pos.y;
466 field.SetPosition( pos );
467 }
468
469 int ordinal = 42; // Arbitrarily larger than any mandatory FIELD_T ids.
470
471 for( SCH_FIELD& field : *m_fields )
472 {
473 if( !field.IsMandatory() )
474 field.SetOrdinal( ordinal++ );
475 }
476
477 for( int ii = m_fields->GetNumberRows() - 1; ii >= 0; ii-- )
478 {
479 SCH_FIELD& field = m_fields->at( ii );
480
481 if( field.IsMandatory() )
482 continue;
483
484 const wxString& fieldName = field.GetCanonicalName();
485
486 if( field.GetText().IsEmpty() )
487 {
488 if( fieldName.IsEmpty() || m_addedTemplateFields.contains( fieldName ) )
489 m_fields->erase( m_fields->begin() + ii );
490 }
491 else if( fieldName.IsEmpty() )
492 {
493 field.SetName( _( "untitled" ) );
494 }
495 }
496
498
499 // Update the parent for inherited symbols
500 if( m_libEntry->IsDerived() )
501 {
502 wxString parentName = EscapeString( m_inheritanceSelectCombo->GetValue(), CTX_LIBID );
503
504 // The parentName was verified to be non-empty in the Validator
505 wxString libName = m_Parent->GetCurLib();
506
507 // Get the parent from the libManager based on the name set in the inheritance combo box.
508 LIB_SYMBOL* newParent = m_Parent->GetLibManager().GetSymbol( parentName, libName );
509
510 // Verify that the requested parent exists
511 wxCHECK( newParent, false );
512
513 m_libEntry->SetParent( newParent );
514 }
515
516 m_libEntry->SetName( newName );
517 m_libEntry->SetKeyWords( m_KeywordCtrl->GetValue() );
520 !m_OptionPartsInterchangeable->GetValue() );
523
524 if( m_OptionPower->GetValue() )
525 {
526 if( m_OptionLocalPower->GetValue() )
528 else
530
531 // Power symbols must have value matching name for now
532 m_libEntry->GetValueField().SetText( newName );
533 }
534 else
535 {
537 }
538
542
545
546 if( m_PinsNameInsideButt->GetValue() )
547 {
548 int offset = KiROUND( (double) m_pinNameOffset.GetValue() );
549
550 // We interpret an offset of 0 as "outside", so make sure it's non-zero
551 m_libEntry->SetPinNameOffset( offset == 0 ? 20 : offset );
552 }
553 else
554 {
555 m_libEntry->SetPinNameOffset( 0 ); // pin text outside the body (name is on the pin)
556 }
557
559
561
562 std::vector<std::set<wxString>>& jumpers = m_libEntry->JumperPinGroups();
563 jumpers.clear();
564
565 for( unsigned i = 0; i < m_listJumperPinGroups->GetCount(); ++i )
566 {
567 wxStringTokenizer tokenizer( m_listJumperPinGroups->GetString( i ), ", " );
568 std::set<wxString>& group = jumpers.emplace_back();
569
570 while( tokenizer.HasMoreTokens() )
571 {
572 if( wxString token = tokenizer.GetNextToken(); !token.IsEmpty() )
573 group.insert( token );
574 }
575 }
576
578
579 // It's possible that the symbol being edited has no pins, in which case there may be no
580 // alternate body style objects causing #LIB_SYMBOL::HasAlternateBodyStyle() to always return
581 // false. This allows the user to edit the alternate body style just in case this condition
582 // occurs.
584
585 return true;
586}
587
588
590{
591 wxGridCellEditor* editor = m_grid->GetCellEditor( event.GetRow(), event.GetCol() );
592 wxControl* control = editor->GetControl();
593
594 if( control && control->GetValidator() && !control->GetValidator()->Validate( control ) )
595 {
596 event.Veto();
597
599 m_delayedFocusRow = event.GetRow();
600 m_delayedFocusColumn = event.GetCol();
602 }
603 else if( event.GetCol() == FDC_NAME )
604 {
605 wxString newName = event.GetString();
606
607 for( int i = 0; i < m_grid->GetNumberRows(); ++i )
608 {
609 if( i == event.GetRow() )
610 continue;
611
612 if( newName.CmpNoCase( m_grid->GetCellValue( i, FDC_NAME ) ) == 0 )
613 {
614 DisplayError( this, wxString::Format( _( "The name '%s' is already in use." ),
615 newName ) );
616 event.Veto();
617 m_delayedFocusRow = event.GetRow();
618 m_delayedFocusColumn = event.GetCol();
619 }
620 }
621 }
622
623 editor->DecRef();
624}
625
626
628{
629 if( m_OptionPower->IsChecked() )
630 {
631 m_grid->SetCellValue( m_fields->GetFieldRow( FIELD_T::VALUE ), FDC_VALUE,
632 m_SymbolNameCtrl->GetValue() );
633 }
634
635 OnModify();
636}
637
638
640{
641 if( !m_delayedFocusCtrl )
642 {
643 // If the validation fails and we throw up a dialog then GTK will give us another
644 // KillFocus event and we end up in infinite recursion. So we use m_delayedFocusCtrl
645 // as a re-entrancy block and then clear it again if validation passes.
648
649 if( m_SymbolNameCtrl->GetValidator()->Validate( m_SymbolNameCtrl ) )
650 {
651 m_delayedFocusCtrl = nullptr;
653 }
654 }
655
656 event.Skip();
657}
658
659
660void DIALOG_LIB_SYMBOL_PROPERTIES::OnAddField( wxCommandEvent& event )
661{
663 return;
664
666 SCH_FIELD newField( m_libEntry, FIELD_T::USER,
668
670 schIUScale.MilsToIU( settings->m_Defaults.text_size ) ) );
671 newField.SetVisible( false );
672
673 m_fields->push_back( newField );
674
675 // notify the grid
676 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, 1 );
677 m_grid->ProcessTableMessage( msg );
678
679 m_grid->MakeCellVisible( (int) m_fields->size() - 1, 0 );
680 m_grid->SetGridCursor( (int) m_fields->size() - 1, 0 );
681
682 m_grid->EnableCellEditControl();
683 m_grid->ShowCellEditControl();
684
685 OnModify();
686}
687
688
690{
691 wxArrayInt selectedRows = m_grid->GetSelectedRows();
692
693 if( selectedRows.empty() && m_grid->GetGridCursorRow() >= 0 )
694 selectedRows.push_back( m_grid->GetGridCursorRow() );
695
696 if( selectedRows.empty() )
697 return;
698
699 for( int row : selectedRows )
700 {
701 if( row < m_fields->GetMandatoryRowCount() )
702 {
703 DisplayError( this, wxString::Format( _( "The first %d fields are mandatory." ),
705 return;
706 }
707 }
708
709 m_grid->CommitPendingChanges( true /* quiet mode */ );
710 m_grid->ClearSelection();
711
712 // Reverse sort so deleting a row doesn't change the indexes of the other rows.
713 selectedRows.Sort( []( int* first, int* second ) { return *second - *first; } );
714
715 for( int row : selectedRows )
716 {
717 m_fields->erase( m_fields->begin() + row );
718
719 // notify the grid
720 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_DELETED, row, 1 );
721 m_grid->ProcessTableMessage( msg );
722
723 if( m_grid->GetNumberRows() > 0 )
724 {
725 m_grid->MakeCellVisible( std::max( 0, row-1 ), m_grid->GetGridCursorCol() );
726 m_grid->SetGridCursor( std::max( 0, row-1 ), m_grid->GetGridCursorCol() );
727 }
728 }
729
730 OnModify();
731}
732
733
734void DIALOG_LIB_SYMBOL_PROPERTIES::OnMoveUp( wxCommandEvent& event )
735{
737 return;
738
739 int i = m_grid->GetGridCursorRow();
740
741 if( i > m_fields->GetMandatoryRowCount() )
742 {
743 SCH_FIELD tmp = m_fields->at( (unsigned) i );
744 m_fields->erase( m_fields->begin() + i, m_fields->begin() + i + 1 );
745 m_fields->insert( m_fields->begin() + i - 1, tmp );
746 m_grid->ForceRefresh();
747
748 m_grid->SetGridCursor( i - 1, m_grid->GetGridCursorCol() );
749 m_grid->MakeCellVisible( m_grid->GetGridCursorRow(), m_grid->GetGridCursorCol() );
750
751 OnModify();
752 }
753 else
754 {
755 wxBell();
756 }
757}
758
759
760void DIALOG_LIB_SYMBOL_PROPERTIES::OnMoveDown( wxCommandEvent& event )
761{
763 return;
764
765 int i = m_grid->GetGridCursorRow();
766
767 if( i >= m_fields->GetMandatoryRowCount() && i + 1 < m_fields->GetNumberRows() )
768 {
769 SCH_FIELD tmp = m_fields->at( (unsigned) i );
770 m_fields->erase( m_fields->begin() + i, m_fields->begin() + i + 1 );
771 m_fields->insert( m_fields->begin() + i + 1, tmp );
772 m_grid->ForceRefresh();
773
774 m_grid->SetGridCursor( i + 1, m_grid->GetGridCursorCol() );
775 m_grid->MakeCellVisible( m_grid->GetGridCursorRow(), m_grid->GetGridCursorCol() );
776
777 OnModify();
778 }
779 else
780 {
781 wxBell();
782 }
783}
784
785
787{
789 return;
790
791 std::vector<SCH_FIELD> fields;
792
793 for( const SCH_FIELD& field : *m_fields )
794 fields.emplace_back( field );
795
796 DIALOG_SIM_MODEL dialog( this, m_parentFrame, *m_libEntry, fields );
797
798 if( dialog.ShowModal() != wxID_OK )
799 return;
800
801 // Add in any new fields
802 for( const SCH_FIELD& editedField : fields )
803 {
804 bool found = false;
805
806 for( SCH_FIELD& existingField : *m_fields )
807 {
808 if( existingField.GetName() == editedField.GetName() )
809 {
810 found = true;
811 existingField.SetText( editedField.GetText() );
812 break;
813 }
814 }
815
816 if( !found )
817 {
818 m_fields->emplace_back( editedField );
819 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, 1 );
820 m_grid->ProcessTableMessage( msg );
821 }
822 }
823
824 // Remove any deleted fields
825 for( int ii = (int) m_fields->size() - 1; ii >= 0; --ii )
826 {
827 SCH_FIELD& existingField = m_fields->at( ii );
828 bool found = false;
829
830 for( SCH_FIELD& editedField : fields )
831 {
832 if( editedField.GetName() == existingField.GetName() )
833 {
834 found = true;
835 break;
836 }
837 }
838
839 if( !found )
840 {
841 m_fields->erase( m_fields->begin() + ii );
842 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_DELETED, ii, 1 );
843 m_grid->ProcessTableMessage( msg );
844 }
845 }
846
847 OnModify();
848 m_grid->ForceRefresh();
849}
850
851
853{
854 int idx = m_FootprintFilterListBox->HitTest( event.GetPosition() );
855 wxCommandEvent dummy;
856
857 if( idx >= 0 )
859 else
861}
862
863
865{
866 // Running the Footprint Browser gums up the works and causes the automatic cancel
867 // stuff to no longer work. So we do it here ourselves.
868 EndQuasiModal( wxID_CANCEL );
869}
870
871
873{
874 wxString filterLine;
875 WX_TEXT_ENTRY_DIALOG dlg( this, _( "Filter:" ), _( "Add Footprint Filter" ), filterLine );
876
877 if( dlg.ShowModal() == wxID_CANCEL || dlg.GetValue().IsEmpty() )
878 return;
879
880 filterLine = dlg.GetValue();
881 filterLine.Replace( wxT( " " ), wxT( "_" ) );
882
883 // duplicate filters do no harm, so don't be a nanny.
884 m_FootprintFilterListBox->Append( filterLine );
885 m_FootprintFilterListBox->SetSelection( (int) m_FootprintFilterListBox->GetCount() - 1 );
886
887 OnModify();
888}
889
890
892{
893 wxArrayInt selections;
894 int n = m_FootprintFilterListBox->GetSelections( selections );
895
896 if( n > 0 )
897 {
898 // Just edit the first one
899 int idx = selections[0];
900 wxString filter = m_FootprintFilterListBox->GetString( idx );
901
902 WX_TEXT_ENTRY_DIALOG dlg( this, _( "Filter:" ), _( "Edit Footprint Filter" ), filter );
903
904 if( dlg.ShowModal() == wxID_OK && !dlg.GetValue().IsEmpty() )
905 {
906 m_FootprintFilterListBox->SetString( (unsigned) idx, dlg.GetValue() );
907 OnModify();
908 }
909 }
910}
911
912
914{
915 // Account for scroll bars
917
918 m_grid->AutoSizeColumn( FDC_NAME );
919 m_grid->SetColSize( FDC_NAME, std::max( 72, m_grid->GetColSize( FDC_NAME ) ) );
920
921 int fixedColsWidth = m_grid->GetColSize( FDC_NAME );
922
923 for( int i = 2; i < m_grid->GetNumberCols(); i++ )
924 fixedColsWidth += m_grid->GetColSize( i );
925
926 m_grid->SetColSize( FDC_VALUE, std::max( 120, width - fixedColsWidth ) );
927}
928
929
930void DIALOG_LIB_SYMBOL_PROPERTIES::OnUpdateUI( wxUpdateUIEvent& event )
931{
932 m_OptionPartsInterchangeable->Enable( m_SelNumberOfUnits->GetValue() > 1 );
934
935 if( m_grid->IsCellEditControlShown() )
936 {
937 int row = m_grid->GetGridCursorRow();
938 int col = m_grid->GetGridCursorCol();
939
940 if( row == m_fields->GetFieldRow( FIELD_T::VALUE ) && col == FDC_VALUE
941 && m_OptionPower->IsChecked() )
942 {
943 wxGridCellEditor* editor = m_grid->GetCellEditor( row, col );
944 m_SymbolNameCtrl->ChangeValue( editor->GetValue() );
945 editor->DecRef();
946 }
947 }
948
949 // Handle shown columns changes
950 std::bitset<64> shownColumns = m_grid->GetShownColumns();
951
952 if( shownColumns != m_shownColumns )
953 {
954 m_shownColumns = shownColumns;
955
956 if( !m_grid->IsCellEditControlShown() )
958 }
959
960 // Handle a delayed focus. The delay allows us to:
961 // a) change focus when the error was triggered from within a killFocus handler
962 // b) show the correct notebook page in the background before the error dialog comes up
963 // when triggered from an OK or a notebook page change
964
965 if( m_delayedFocusPage >= 0 && m_NoteBook->GetSelection() != m_delayedFocusPage )
966 {
967 m_NoteBook->ChangeSelection( (unsigned) m_delayedFocusPage );
969 }
970
971 if( !m_delayedErrorMessage.IsEmpty() )
972 {
973 // We will re-enter this routine when the error dialog is displayed, so make
974 // sure we don't keep putting up more dialogs.
975 wxString msg = m_delayedErrorMessage;
976 m_delayedErrorMessage = wxEmptyString;
977
978 // Do not use DisplayErrorMessage(); it screws up window order on Mac
979 DisplayError( nullptr, msg );
980 }
981
983 {
984 m_delayedFocusCtrl->SetFocus();
985
986 if( auto textEntry = dynamic_cast<wxTextEntry*>( m_delayedFocusCtrl ) )
987 textEntry->SelectAll();
988
989 m_delayedFocusCtrl = nullptr;
990 }
991 else if( m_delayedFocusGrid )
992 {
993 m_delayedFocusGrid->SetFocus();
996
997 m_delayedFocusGrid->EnableCellEditControl( true );
998 m_delayedFocusGrid->ShowCellEditControl();
999
1000 m_delayedFocusGrid = nullptr;
1001 m_delayedFocusRow = -1;
1003 }
1004}
1005
1006
1008{
1009 auto new_size = event.GetSize();
1010
1011 if( new_size != m_size )
1012 {
1013 m_size = new_size;
1014
1016 }
1017
1018 // Always propagate a wxSizeEvent:
1019 event.Skip();
1020}
1021
1022
1024{
1025 bSizerLowerBasicPanel->Show( !aIsAlias );
1026 m_inheritanceSelectCombo->Enable( aIsAlias );
1027 m_inheritsStaticText->Enable( aIsAlias );
1028 m_grid->ForceRefresh();
1029}
1030
1031
1033{
1034 if( m_OptionPower->IsChecked() )
1035 {
1036 m_excludeFromSimCheckBox->SetValue( true );
1037 m_excludeFromBomCheckBox->SetValue( true );
1038 m_excludeFromBoardCheckBox->SetValue( true );
1039 m_excludeFromBomCheckBox->Enable( false );
1040 m_excludeFromBoardCheckBox->Enable( false );
1041 m_excludeFromSimCheckBox->Enable( false );
1042 m_spiceFieldsButton->Show( false );
1043 m_OptionLocalPower->Enable( true );
1044 }
1045 else
1046 {
1047 m_excludeFromBomCheckBox->Enable( true );
1048 m_excludeFromBoardCheckBox->Enable( true );
1049 m_excludeFromSimCheckBox->Enable( true );
1050 m_spiceFieldsButton->Show( true );
1051 m_OptionLocalPower->Enable( false );
1052 }
1053
1054 OnModify();
1055}
1056
1057
1058void DIALOG_LIB_SYMBOL_PROPERTIES::OnText( wxCommandEvent& event )
1059{
1060 OnModify();
1061}
1062
1063
1064void DIALOG_LIB_SYMBOL_PROPERTIES::OnCombobox( wxCommandEvent& event )
1065{
1066 OnModify();
1067}
1068
1069
1070void DIALOG_LIB_SYMBOL_PROPERTIES::OnCheckBox( wxCommandEvent& event )
1071{
1072 OnModify();
1073}
1074
1075
1077{
1078 OnModify();
1079}
1080
1081
1083{
1084 OnModify();
1085}
1086
1087
1089{
1091 aEvent.Veto();
1092}
1093
1094
1096{
1097 wxArrayInt selections;
1098 int n = m_listAvailablePins->GetSelections( selections );
1099 wxCHECK( n > 0, /* void */ );
1100
1101 m_listJumperPinGroups->Freeze();
1102 m_listAvailablePins->Freeze();
1103
1104 wxString group;
1105 int i = 0;
1106
1107 for( int idx : selections )
1108 {
1109 group << m_listAvailablePins->GetString( idx );
1110
1111 if( ++i < n )
1112 group << ", ";
1113 }
1114
1115 for( int idx = selections.size() - 1; idx >= 0; --idx )
1116 m_listAvailablePins->Delete( selections[idx] );
1117
1118 m_listJumperPinGroups->AppendString( group );
1119
1120 m_listJumperPinGroups->Thaw();
1121 m_listAvailablePins->Thaw();
1122}
1123
1124
1126{
1127 wxArrayInt selections;
1128 int n = m_listJumperPinGroups->GetSelections( selections );
1129 wxCHECK( n > 0, /* void */ );
1130
1131 m_listJumperPinGroups->Freeze();
1132 m_listAvailablePins->Freeze();
1133
1134 for( int idx : selections )
1135 {
1136 wxStringTokenizer tokenizer( m_listJumperPinGroups->GetString( idx ), ", " );
1137
1138 while( tokenizer.HasMoreTokens() )
1139 {
1140 if( wxString token = tokenizer.GetNextToken(); !token.IsEmpty() )
1141 m_listAvailablePins->AppendString( token );
1142 }
1143 }
1144
1145 for( int idx = selections.size() - 1; idx >= 0; --idx )
1146 m_listJumperPinGroups->Delete( selections[idx] );
1147
1148 m_listJumperPinGroups->Thaw();
1149 m_listAvailablePins->Thaw();
1150}
1151
1152
1154{
1155 wxArrayInt selections;
1156 int n = m_listJumperPinGroups->GetSelections( selections );
1157 m_btnRemoveJumperPinGroup->Enable( n > 0 );
1158}
1159
1160
1162{
1163 wxArrayInt selections;
1164 int n = m_listAvailablePins->GetSelections( selections );
1165 m_btnCreateJumperPinGroup->Enable( n > 0 );
1166}
constexpr EDA_IU_SCALE schIUScale
Definition: base_units.h:110
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition: bitmap.cpp:110
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
Class DIALOG_LIB_SYMBOL_PROPERTIES_BASE.
void OnSymbolNameKillFocus(wxFocusEvent &event) override
std::unique_ptr< LISTBOX_TRICKS > m_fpFilterTricks
void OnPageChanging(wxNotebookEvent &event) override
void OnSpinCtrlText(wxCommandEvent &event) override
void OnSizeGrid(wxSizeEvent &event) override
DIALOG_LIB_SYMBOL_PROPERTIES(SYMBOL_EDIT_FRAME *parent, LIB_SYMBOL *aLibEntry)
void OnGroupedPinListClick(wxCommandEvent &event) override
void OnUpdateUI(wxUpdateUIEvent &event) override
void OnCancelButtonClick(wxCommandEvent &event) override
void OnEditFootprintFilter(wxCommandEvent &event) override
void OnDeleteField(wxCommandEvent &event) override
void OnCombobox(wxCommandEvent &event) override
void OnFpFilterDClick(wxMouseEvent &event) override
void OnMoveUp(wxCommandEvent &event) override
void OnAvailablePinsClick(wxCommandEvent &event) override
void OnText(wxCommandEvent &event) override
void OnMoveDown(wxCommandEvent &event) override
void OnAddFootprintFilter(wxCommandEvent &event) override
void OnCheckBox(wxCommandEvent &event) override
void OnSymbolNameText(wxCommandEvent &event) override
void onPowerCheckBox(wxCommandEvent &aEvent) override
void OnBtnCreateJumperPinGroup(wxCommandEvent &event) override
void OnAddField(wxCommandEvent &event) override
void OnEditSpiceModel(wxCommandEvent &event) override
void OnSpinCtrl(wxSpinEvent &event) override
void OnBtnRemoveJumperPinGroup(wxCommandEvent &event) override
void SetupStandardButtons(std::map< int, wxString > aLabels={})
void resetSize()
Clear the existing dialog size and position.
void EndQuasiModal(int retCode)
void OnModify()
void finishDialogSettings()
In all dialogs, we must call the same functions to fix minimal dlg size, the default position and per...
EDA_BASE_FRAME * m_parentFrame
Definition: dialog_shim.h:212
int ShowModal() override
void SetTextSize(VECTOR2I aNewSize, bool aEnforceMinTextSize=true)
Definition: eda_text.cpp:526
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition: eda_text.h:98
virtual void SetVisible(bool aVisible)
Definition: eda_text.cpp:379
SCH_FIELD * GetField(FIELD_T aFieldId)
int GetNumberRows() override
int GetMandatoryRowCount() const
int GetFieldRow(FIELD_T aFieldId)
A text control validator used for validating the text allowed in fields.
Definition: validators.h:142
Define a library symbol object.
Definition: lib_symbol.h:85
wxString GetKeyWords() const override
Definition: lib_symbol.h:183
void SetGlobalPower()
Definition: lib_symbol.cpp:473
std::vector< SCH_PIN * > GetPins(int aUnit, int aBodyStyle) const
Return a list of pin object pointers from the draw item list.
Definition: lib_symbol.cpp:797
bool IsPower() const override
Definition: lib_symbol.cpp:467
void SetUnitCount(int aCount, bool aDuplicateDrawItems=true)
Set the units per symbol count.
bool UnitsLocked() const
Check whether symbol units are interchangeable.
Definition: lib_symbol.h:289
bool IsRoot() const override
For symbols derived from other symbols, IsRoot() indicates no derivation.
Definition: lib_symbol.h:206
bool IsDerived() const
Definition: lib_symbol.h:207
void SetDuplicatePinNumbersAreJumpers(bool aEnabled)
Definition: lib_symbol.h:571
void SetFields(const std::vector< SCH_FIELD > &aFieldsList)
Overwrite all the existing fields in this symbol with fields supplied in aFieldsList.
Definition: lib_symbol.cpp:997
bool HasAlternateBodyStyle() const override
Test if symbol has more than one body conversion type (DeMorgan).
void LockUnits(bool aLockUnits)
Set interchangeable the property for symbol units.
Definition: lib_symbol.h:283
void SetParent(LIB_SYMBOL *aParent=nullptr)
Definition: lib_symbol.cpp:333
wxString GetName() const override
Definition: lib_symbol.h:149
void SetKeyWords(const wxString &aKeyWords)
Definition: lib_symbol.h:181
SCH_FIELD & GetValueField()
Return reference to the value field.
Definition: lib_symbol.h:334
bool IsLocalPower() const override
Definition: lib_symbol.cpp:421
wxArrayString GetFPFilters() const
Definition: lib_symbol.h:218
void SetLocalPower()
Definition: lib_symbol.cpp:437
void SetFPFilters(const wxArrayString &aFilters)
Definition: lib_symbol.h:216
void CopyFields(std::vector< SCH_FIELD > &aList)
Create a copy of the SCH_FIELDs, sorted in ordinal order.
void SetHasAlternateBodyStyle(bool aHasAlternate, bool aDuplicatePins=true)
Set or clear the alternate body style (DeMorgan) for the symbol.
int GetUnitCount() const override
bool GetDuplicatePinNumbersAreJumpers() const
Definition: lib_symbol.h:570
LIB_SYMBOL_REF & GetParent()
Definition: lib_symbol.h:118
std::vector< std::set< wxString > > & JumperPinGroups()
Each jumper pin group is a set of pin numbers that should be treated as internally connected.
Definition: lib_symbol.h:577
virtual void SetName(const wxString &aName)
Definition: lib_symbol.cpp:326
void SetNormal()
Definition: lib_symbol.cpp:501
EMBEDDED_FILES * GetLocalFiles()
bool TransferDataToWindow() override
virtual SETTINGS_MANAGER & GetSettingsManager() const
Definition: pgm_base.h:125
bool IsMandatory() const
Definition: sch_field.cpp:1336
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 SetName(const wxString &aName)
Definition: sch_field.cpp:1059
void SetText(const wxString &aText) override
Definition: sch_field.cpp:1069
T * GetAppSettings(const char *aFilename)
Return a handle to the a given settings by type.
void SetBitmap(const wxBitmapBundle &aBmp)
The symbol library editor main window.
void UpdateAfterSymbolProperties(wxString *aOldName=nullptr)
wxString GetCurLib() const
The nickname of the current library being edited and empty string if none.
bool GetShowDeMorgan() const
bool IsSymbolFromLegacyLibrary() const
bool IsSymbolFromSchematic() const
void SaveCopyInUndoList(const wxString &aDescription, LIB_SYMBOL *aSymbol, UNDO_REDO aUndoType=UNDO_REDO::LIBEDIT)
Create a copy of the current symbol, and save it in the undo list.
SYMBOL_EDITOR_SETTINGS * GetSettings() const
LIB_SYMBOL_LIBRARY_MANAGER & GetLibManager()
void SetShowDeMorgan(bool show)
bool SymbolNameInUse(const wxString &aName, const wxString &aLibrary)
Return true if the symbol name is already in use in the specified library.
LIB_SYMBOL * GetSymbol(const wxString &aSymbolName, const wxString &aLibrary) const
Return either an alias of a working LIB_SYMBOL copy, or alias of the original symbol if there is no w...
void GetSymbolNames(const wxString &aLibName, wxArrayString &aSymbolNames, SYMBOL_NAME_FILTER aFilter=SYMBOL_NAME_FILTER::ALL)
bool GetExcludedFromBoard() const
Definition: symbol.h:187
virtual void SetShowPinNumbers(bool aShow)
Set or clear the pin number visibility flag.
Definition: symbol.h:164
bool GetExcludedFromBOM() const
Definition: symbol.h:181
void SetExcludedFromSim(bool aExcludeFromSim) override
Set or clear the exclude from simulation flag.
Definition: symbol.h:170
int GetPinNameOffset() const
Definition: symbol.h:153
virtual void SetShowPinNames(bool aShow)
Set or clear the pin name visibility flag.
Definition: symbol.h:158
virtual bool GetShowPinNames() const
Definition: symbol.h:159
void SetExcludedFromBOM(bool aExcludeFromBOM)
Set or clear the exclude from schematic bill of materials flag.
Definition: symbol.h:180
void SetPinNameOffset(int aOffset)
Set the offset in mils of the pin name text from the pin symbol.
Definition: symbol.h:152
void SetExcludedFromBoard(bool aExcludeFromBoard)
Set or clear exclude from board netlist flag.
Definition: symbol.h:186
virtual bool GetShowPinNumbers() const
Definition: symbol.h:165
bool GetExcludedFromSim() const override
Definition: symbol.h:175
void AddTemplateFieldNames(const wxString &aSerializedFieldNames)
Add a serialized list of template field names.
const std::vector< TEMPLATE_FIELDNAME > & GetTemplateFieldNames()
Return a template field name list for read only access.
virtual long long int GetValue()
Return the current value in Internal Units.
void Enable(bool aEnable)
Enable/disable the label, widget and units label.
virtual void ChangeValue(int aValue)
Set new value (in Internal Units) for the text field, taking care of units conversion WITHOUT trigger...
void ShowHideColumns(const wxString &shownColumns)
Show/hide the grid columns based on a tokenized string of shown column indexes.
Definition: wx_grid.cpp:494
void SetTable(wxGridTableBase *table, bool aTakeOwnership=false)
Hide wxGrid's SetTable() method with one which doesn't mess up the grid column widths when setting th...
Definition: wx_grid.cpp:275
void DestroyTable(wxGridTableBase *aTable)
Work-around for a bug in wxGrid which crashes when deleting the table if the cell edit control was no...
Definition: wx_grid.cpp:449
wxString GetShownColumnsAsString()
Get a tokenized string containing the shown column indexes.
Definition: wx_grid.cpp:464
std::bitset< 64 > GetShownColumns()
Definition: wx_grid.cpp:483
bool CommitPendingChanges(bool aQuietMode=false)
Close any open cell edit controls.
Definition: wx_grid.cpp:644
A KICAD version of wxTextEntryDialog which supports the various improvements/work-arounds from DIALOG...
wxString GetValue() const
bool IsOK(wxWindow *aParent, const wxString &aMessage)
Display a yes/no dialog with aMessage and returns the user response.
Definition: confirm.cpp:249
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition: confirm.cpp:194
void DisplayError(wxWindow *aParent, const wxString &aText)
Display an error or warning message box with aMessage.
Definition: confirm.cpp:170
This file is part of the common library.
#define _(s)
@ FDC_NAME
@ FDC_VALUE
std::shared_ptr< LIB_SYMBOL > LIB_SYMBOL_SPTR
shared pointer to LIB_SYMBOL
Definition: lib_symbol.h:52
wxSize GetUnobscuredSize(const wxWindow *aWindow)
Tries to determine the size of the viewport of a scrollable widget (wxDataViewCtrl,...
Definition: wxgtk/ui.cpp:252
wxString GetRefDesPrefix(const wxString &aRefDes)
Get the (non-numeric) prefix from a refdes - e.g.
STL namespace.
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
Collection of utility functions for component reference designators (refdes)
std::vector< FAB_LAYER_COLOR > dummy
int StrNumCmp(const wxString &aString1, const wxString &aString2, bool aIgnoreCase)
Compare two strings with alphanumerical content.
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:...
@ CTX_LIBID
Definition: string_utils.h:54
constexpr int MilsToIU(int mils) const
Definition: base_units.h:93
Hold a name of a symbol's field, field value, and default visibility.
wxString GetUserFieldName(int aFieldNdx, bool aTranslateForHI)
#define DO_TRANSLATE
UNDO_REDO
Undo Redo considerations: Basically we have 3 cases New item Deleted item Modified item there is also...
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695