KiCad PCB EDA Suite
Loading...
Searching...
No Matches
footprint_libraries_utils.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
24#include <memory>
25#include <wx/ffile.h>
26#include <pgm_base.h>
27#include <kiface_base.h>
28#include <confirm.h>
29#include <kidialog.h>
30#include <macros.h>
31#include <pcb_edit_frame.h>
32#include <eda_list_dialog.h>
33#include <filter_reader.h>
35#include <validators.h>
37#include <tool/tool_manager.h>
38#include <tools/pcb_actions.h>
40#include <tools/pad_tool.h>
41#include <footprint.h>
42#include <zone.h>
43#include <pcb_group.h>
48#include <env_paths.h>
49#include <paths.h>
51#include <kiplatform/ui.h>
52#include <project_pcb.h>
57#include <view/view_controls.h>
58#include <wx/filedlg.h>
59#include <wx/fswatcher.h>
60
61
62static constexpr int ID_MAKE_NEW_LIBRARY = 4173;
63
64
65// unique, "file local" translations:
66
67
68static const wxString INFO_LEGACY_LIB_WARN_EDIT(
69 _( "Writing/modifying legacy libraries (.mod files) is not allowed\n"\
70 "Please save the current library to the new .pretty format\n"\
71 "and update your footprint lib table\n"\
72 "to save your footprint (a .kicad_mod file) in the .pretty library folder" ) );
73
74static const wxString INFO_LEGACY_LIB_WARN_DELETE(
75 _( "Modifying legacy libraries (.mod files) is not allowed\n"\
76 "Please save the current library under the new .pretty format\n"\
77 "and update your footprint lib table\n"\
78 "before deleting a footprint" ) );
79
80
82{
83 wxFileName fn;
84
85 if( !aName.empty() )
86 {
87 fn = aName;
88 }
89 else
90 {
91 // Prompt the user for a footprint file to open.
92 static int lastFilterIndex = 0; // To store the last choice during a session.
93 wxString fileFiltersStr;
94 std::vector<std::string> allExtensions;
95 std::set<wxString> allWildcardsSet;
96
97 for( const auto& plugin : PCB_IO_MGR::PLUGIN_REGISTRY::Instance()->AllPlugins() )
98 {
99 IO_RELEASER<PCB_IO> pi( plugin.m_createFunc() );
100
101 if( !pi )
102 continue;
103
104 const IO_BASE::IO_FILE_DESC& desc = pi->GetLibraryFileDesc();
105
106 if( !desc )
107 continue;
108
109 if( !fileFiltersStr.IsEmpty() )
110 fileFiltersStr += wxChar( '|' );
111
112 fileFiltersStr += desc.FileFilter();
113
114 for( const std::string& ext : desc.m_FileExtensions )
115 {
116 allExtensions.emplace_back( ext );
117 allWildcardsSet.insert( wxT( "*." ) + formatWildcardExt( ext ) + wxT( ";" ) );
118 }
119 }
120
121 wxString allWildcardsStr;
122
123 for( const wxString& wildcard : allWildcardsSet )
124 allWildcardsStr << wildcard;
125
126 fileFiltersStr = _( "All supported formats" ) + wxT( "|" ) + allWildcardsStr + wxT( "|" )
127 + fileFiltersStr;
128
129 wxFileDialog dlg( this, _( "Import Footprint" ), m_mruPath, wxEmptyString, fileFiltersStr,
130 wxFD_OPEN | wxFD_FILE_MUST_EXIST );
131
132 wxArrayString dummy1, dummy2;
133 const int nWildcards = wxParseCommonDialogsFilter( fileFiltersStr, dummy1, dummy2 );
134
135 if( lastFilterIndex >= 0 && lastFilterIndex < nWildcards )
136 dlg.SetFilterIndex( lastFilterIndex );
137
139
140 if( dlg.ShowModal() == wxID_CANCEL )
141 return nullptr;
142
143 lastFilterIndex = dlg.GetFilterIndex();
144
145 fn = dlg.GetPath();
146 }
147
148 if( !fn.IsOk() )
149 return nullptr;
150
151 if( !wxFileExists( fn.GetFullPath() ) )
152 {
153 wxString msg = wxString::Format( _( "File '%s' not found." ), fn.GetFullPath() );
154 DisplayError( this, msg );
155 return nullptr;
156 }
157
158 m_mruPath = fn.GetPath();
159
161
162 for( const auto& plugin : PCB_IO_MGR::PLUGIN_REGISTRY::Instance()->AllPlugins() )
163 {
164 IO_RELEASER<PCB_IO> pi( plugin.m_createFunc() );
165
166 if( !pi )
167 continue;
168
169 if( pi->GetLibraryFileDesc().m_FileExtensions.empty() )
170 continue;
171
172 if( pi->CanReadFootprint( fn.GetFullPath() ) )
173 {
174 fileType = plugin.m_type;
175 break;
176 }
177 }
178
180 {
181 DisplayError( this, _( "Not a footprint file." ) );
182 return nullptr;
183 }
184
185 FOOTPRINT* footprint = nullptr;
186 wxString footprintName;
187
188 try
189 {
191
192 footprint = pi->ImportFootprint( fn.GetFullPath(), footprintName);
193
194 if( !footprint )
195 {
196 wxString msg = wxString::Format( _( "Unable to load footprint '%s' from '%s'" ),
197 footprintName, fn.GetFullPath() );
198 DisplayError( this, msg );
199 return nullptr;
200 }
201 }
202 catch( const IO_ERROR& ioe )
203 {
204 DisplayError( this, ioe.What() );
205
206 // if the footprint is not loaded, exit.
207 // However, even if an error happens, it can be loaded, because in KICAD and GPCB format,
208 // a fp library is a set of separate files, and the error(s) are not necessary when
209 // reading the selected file
210
211 if( !footprint )
212 return nullptr;
213 }
214
215 footprint->SetFPID( LIB_ID( wxEmptyString, footprintName ) );
216
217 // Insert footprint in list
218 AddFootprintToBoard( footprint );
219
220 // Display info :
221 SetMsgPanel( footprint );
222 PlaceFootprint( footprint );
223
224 footprint->SetPosition( VECTOR2I( 0, 0 ) );
225
227 UpdateView();
228
229 return footprint;
230}
231
232
234{
235 wxFileName fn;
237
238 if( !aFootprint )
239 return;
240
241 fn.SetName( aFootprint->GetFPID().GetLibItemName() );
242
243 wxString wildcard = FILEEXT::KiCadFootprintLibFileWildcard();
244
246
247 if( !cfg->m_LastExportPath.empty() )
248 fn.SetPath( cfg->m_LastExportPath );
249 else
250 fn.SetPath( m_mruPath );
251
252 wxFileDialog dlg( this, _( "Export Footprint" ), fn.GetPath(), fn.GetFullName(),
253 wildcard, wxFD_SAVE | wxFD_OVERWRITE_PROMPT );
254
256
257 if( dlg.ShowModal() == wxID_CANCEL )
258 return;
259
261 cfg->m_LastExportPath = fn.GetPath();
262
263 try
264 {
265 // Export as *.kicad_pcb format, using a strategy which is specifically chosen
266 // as an example on how it could also be used to send it to the system clipboard.
267
269
270 /* This footprint should *already* be "normalized" in a way such that
271 orientation is zero, etc., since it came from the Footprint Editor.
272
273 aFootprint->SetParent( 0 );
274 aFootprint->SetOrientation( 0 );
275 */
276
277 pcb_io.Format( aFootprint );
278
279 FILE* fp = wxFopen( dlg.GetPath(), wxT( "wt" ) );
280
281 if( fp == nullptr )
282 {
283 DisplayErrorMessage( this, wxString::Format( _( "Insufficient permissions to write file '%s'." ),
284 dlg.GetPath() ) );
285 return;
286 }
287
288 std::string prettyData = pcb_io.GetStringOutput( false );
289 KICAD_FORMAT::Prettify( prettyData, KICAD_FORMAT::FORMAT_MODE::NORMAL );
290
291 fprintf( fp, "%s", prettyData.c_str() );
292 fclose( fp );
293 }
294 catch( const IO_ERROR& ioe )
295 {
296 DisplayError( this, ioe.What() );
297 return;
298 }
299
300 wxString msg = wxString::Format( _( "Footprint exported to file '%s'." ), dlg.GetPath() );
301 DisplayInfoMessage( this, msg );
302}
303
304
305wxString PCB_BASE_EDIT_FRAME::CreateNewProjectLibrary( const wxString& aDialogTitle, const wxString& aLibName )
306{
307 return createNewLibrary( aDialogTitle, aLibName, wxEmptyString, LIBRARY_TABLE_SCOPE::PROJECT );
308}
309
310
311wxString PCB_BASE_EDIT_FRAME::CreateNewLibrary( const wxString& aDialogTitle, const wxString& aInitialPath )
312{
313 return createNewLibrary( aDialogTitle, wxEmptyString, aInitialPath );
314}
315
316
317wxString PCB_BASE_EDIT_FRAME::createNewLibrary( const wxString& aDialogTitle, const wxString& aLibName,
318 const wxString& aInitialPath, std::optional<LIBRARY_TABLE_SCOPE> aScope )
319{
320 // Kicad cannot write legacy format libraries, only .pretty new format because the legacy
321 // format cannot handle current features.
322 // The footprint library is actually a directory.
323
324 wxFileName fn;
325 bool doAdd = false;
326 bool isGlobal = false;
327 FILEDLG_HOOK_NEW_LIBRARY tableChooser( isGlobal );
328 FILEDLG_HOOK_NEW_LIBRARY* fileDlgHook = &tableChooser;
329
330 if( aScope )
331 fileDlgHook = nullptr;
332
333 if( aLibName.IsEmpty() )
334 {
335 fn = aInitialPath.IsEmpty() ? Prj().GetProjectPath() : aInitialPath;
336
337 if( !LibraryFileBrowser( aDialogTitle, false, fn, FILEEXT::KiCadFootprintLibPathWildcard(),
338 FILEEXT::KiCadFootprintLibPathExtension, false, fileDlgHook ) )
339 {
340 return wxEmptyString;
341 }
342
343 if( fileDlgHook )
344 {
345 isGlobal = fileDlgHook->GetUseGlobalTable();
347 }
348
349 doAdd = true;
350 }
351 else
352 {
354
355 if( !fn.IsAbsolute() )
356 {
357 fn.SetName( aLibName );
358 fn.MakeAbsolute( Prj().GetProjectPath() );
359 }
360 }
361
362 // We can save fp libs only using PCB_IO_MGR::KICAD_SEXP format (.pretty libraries)
364 wxString libPath = fn.GetFullPath();
365
366 try
367 {
369
370 bool writable = false;
371 bool exists = false;
372
373 try
374 {
375 writable = pi->IsLibraryWritable( libPath );
376 exists = true; // no exception was thrown, lib must exist.
377 }
378 catch( const IO_ERROR& )
379 {
380 // best efforts....
381 }
382
383 if( exists )
384 {
385 if( !writable )
386 {
387 wxString msg = wxString::Format( _( "Library %s is read only." ), libPath );
388 ShowInfoBarError( msg );
389 return wxEmptyString;
390 }
391 else
392 {
393 wxString msg = wxString::Format( _( "Library %s already exists." ), libPath );
394 KIDIALOG dlg( this, msg, _( "Confirmation" ), wxOK | wxCANCEL | wxICON_WARNING );
395 dlg.SetOKLabel( _( "Overwrite" ) );
396 dlg.DoNotShowCheckbox( __FILE__, __LINE__ );
397
398 if( dlg.ShowModal() == wxID_CANCEL )
399 return wxEmptyString;
400
401 pi->DeleteLibrary( libPath );
402 }
403 }
404
405 pi->CreateLibrary( libPath );
406 }
407 catch( const IO_ERROR& ioe )
408 {
409 DisplayError( this, ioe.What() );
410 return wxEmptyString;
411 }
412
413 if( doAdd )
414 AddLibrary( aDialogTitle, libPath, aScope );
415
416 return libPath;
417}
418
419
420wxString PCB_BASE_EDIT_FRAME::SelectLibrary( const wxString& aDialogTitle, const wxString& aListLabel,
421 const std::vector<std::pair<wxString, bool*>>& aExtraCheckboxes )
422{
423 // Keep asking the user for a new name until they give a valid one or cancel the operation
424 while( true )
425 {
426 wxArrayString headers;
427 std::vector<wxArrayString> itemsToDisplay;
428
429 GetLibraryItemsForListDialog( headers, itemsToDisplay );
430
431 wxString libraryName = Prj().GetRString( PROJECT::PCB_LIB_NICKNAME );
432
433 EDA_LIST_DIALOG dlg( this, aDialogTitle, headers, itemsToDisplay, libraryName, false );
434 dlg.SetListLabel( aListLabel );
435
436 for( const auto& [label, val] : aExtraCheckboxes )
437 dlg.AddExtraCheckbox( label, val );
438
439 wxButton* newLibraryButton = new wxButton( &dlg, ID_MAKE_NEW_LIBRARY, _( "New Library..." ) );
440 dlg.m_ButtonsSizer->Prepend( 80, 20 );
441 dlg.m_ButtonsSizer->Prepend( newLibraryButton, 0, wxALIGN_CENTER_VERTICAL|wxLEFT|wxRIGHT, 10 );
442
443 newLibraryButton->Bind( wxEVT_BUTTON,
444 [&dlg]( wxCommandEvent& )
445 {
446 dlg.EndModal( ID_MAKE_NEW_LIBRARY );
448
449 dlg.Layout();
450 dlg.GetSizer()->Fit( &dlg );
451
452 int ret = dlg.ShowModal();
453
454 switch( ret )
455 {
456 case wxID_CANCEL:
457 return wxEmptyString;
458
459 case wxID_OK:
460 libraryName = dlg.GetTextSelection();
461 Prj().SetRString( PROJECT::PCB_LIB_NICKNAME, libraryName );
463 return libraryName;
464
466 {
467 wxFileName fn = CreateNewLibrary( _( "New Footprint Library" ),
468 Prj().GetRString( PROJECT::PCB_LIB_PATH ) );
469
470 Prj().SetRString( PROJECT::PCB_LIB_PATH, fn.GetPath() );
471 Prj().SetRString( PROJECT::PCB_LIB_NICKNAME, fn.GetName() );
472 break;
473 }
474
475 default:
476 break;
477 }
478 }
479}
480
481
482bool PCB_BASE_EDIT_FRAME::AddLibrary( const wxString& aDialogTitle, const wxString& aFilename,
483 std::optional<LIBRARY_TABLE_SCOPE> aScope )
484{
487 bool isGlobal = false;
488 FILEDLG_HOOK_NEW_LIBRARY tableChooser( isGlobal );
489 FILEDLG_HOOK_NEW_LIBRARY* fileDlgHook = &tableChooser;
490
491 if( aScope )
492 {
493 isGlobal = ( *aScope == LIBRARY_TABLE_SCOPE::GLOBAL );
494 fileDlgHook = nullptr;
495 }
496
497 wxFileName fn( aFilename );
498
499 if( aFilename.IsEmpty() )
500 {
501 if( !LibraryFileBrowser( aDialogTitle, true, fn, FILEEXT::KiCadFootprintLibPathWildcard(),
502 FILEEXT::KiCadFootprintLibPathExtension, true, fileDlgHook ) )
503 {
504 return false;
505 }
506
507 if( fileDlgHook )
508 isGlobal = fileDlgHook->GetUseGlobalTable();
509 }
510
512
513 wxString libPath = fn.GetFullPath();
514 wxString libName = fn.GetName();
515
516 if( libName.IsEmpty() )
517 return false;
518
520
521 if( lib_type == PCB_IO_MGR::FILE_TYPE_NONE )
522 lib_type = PCB_IO_MGR::KICAD_SEXP;
523
524 wxString type = PCB_IO_MGR::ShowType( lib_type );
525
526 // KiCad lib is our default guess. So it might not have the .pretty extension
527 // In this case, the extension is part of the library name
528 if( lib_type == PCB_IO_MGR::KICAD_SEXP && fn.GetExt() != FILEEXT::KiCadFootprintLibPathExtension )
529 libName = fn.GetFullName();
530
531 // try to use path normalized to an environmental variable or project path
532 wxString normalizedPath = NormalizePath( libPath, &Pgm().GetLocalEnvVariables(), &Prj() );
533 bool success = true;
534
535 try
536 {
537 std::optional<LIBRARY_TABLE*> optTable = manager.Table( LIBRARY_TABLE_TYPE::FOOTPRINT, aScope.value() );
538 wxCHECK( optTable, false );
539 LIBRARY_TABLE* table = optTable.value();
540
541 LIBRARY_TABLE_ROW& row = table->InsertRow();
542
543 row.SetNickname( libName );
544 row.SetURI( normalizedPath );
545 row.SetType( type );
546
547 table->Save().map_error(
548 [&]( const LIBRARY_ERROR& aError )
549 {
550 wxMessageBox( _( "Error saving library table:\n\n" ) + aError.message,
551 _( "File Save Error" ), wxOK | wxICON_ERROR );
552 success = false;
553 } );
554 }
555 catch( const IO_ERROR& ioe )
556 {
557 DisplayError( this, ioe.What() );
558 return false;
559 }
560
561 if( success )
562 {
563 manager.ReloadTables( aScope.value(), { LIBRARY_TABLE_TYPE::FOOTPRINT } );
564 adapter->LoadOne( fn.GetName() );
565
566 // Don't use dynamic_cast; it will fail across compile units on MacOS
568 {
569 LIB_ID libID( libName, wxEmptyString );
570 editor->SyncLibraryTree( true );
571 editor->FocusOnLibID( libID );
572 }
573
574 auto viewer = (FOOTPRINT_VIEWER_FRAME*) Kiway().Player( FRAME_FOOTPRINT_VIEWER, false );
575
576 if( viewer )
577 viewer->ReCreateLibraryList();
578 }
579
580 return success;
581}
582
583
585{
586 if( !aFPID.IsValid() )
587 return false;
588
591
592 wxString nickname = aFPID.GetLibNickname();
593 wxString fpname = aFPID.GetLibItemName();
594 wxString libfullname;
595
596 // Legacy libraries are readable, but modifying legacy format is not allowed
597 // So prompt the user if he try to delete a footprint from a legacy lib
598 if( std::optional<wxString> optUri = manager.GetFullURI( LIBRARY_TABLE_TYPE::FOOTPRINT, nickname ) )
599 libfullname = *optUri;
600 else
601 return false;
602
604 {
606 return false;
607 }
608
609 if( !adapter->IsFootprintLibWritable( nickname ) )
610 {
611 wxString msg = wxString::Format( _( "Library '%s' is read only." ), nickname );
612 ShowInfoBarError( msg );
613 return false;
614 }
615
616 // Confirmation
617 wxString msg = wxString::Format( _( "Delete footprint '%s' from library '%s'?" ),
618 fpname.GetData(),
619 nickname.GetData() );
620
621 if( aConfirm && !IsOK( this, msg ) )
622 return false;
623
624 try
625 {
626 adapter->DeleteFootprint( nickname, fpname );
627 }
628 catch( const IO_ERROR& ioe )
629 {
630 DisplayError( this, ioe.What() );
631 return false;
632 }
633
634 msg.Printf( _( "Footprint '%s' deleted from library '%s'" ),
635 fpname.GetData(),
636 nickname.GetData() );
637
638 SetStatusText( msg );
639
640 return true;
641}
642
643
644void PCB_EDIT_FRAME::ExportFootprintsToLibrary( bool aStoreInNewLib, const wxString& aLibName,
645 wxString* aLibPath )
646{
647 if( GetBoard()->GetFirstFootprint() == nullptr )
648 {
649 DisplayInfoMessage( this, _( "No footprints to export!" ) );
650 return;
651 }
652
653 bool map = false;
654 PROJECT& prj = Prj();
655 wxString nickname = SelectLibrary( _( "Export Footprints" ), _( "Export footprints to library:" ),
656 { { _( "Update board footprints to link to exported footprints" ), &map } } );
657
658 if( !nickname ) // Aborted
659 return;
660
661 prj.SetRString( PROJECT::PCB_LIB_NICKNAME, nickname );
662
663 for( FOOTPRINT* footprint : GetBoard()->Footprints() )
664 {
665 try
666 {
668
669 if( !footprint->GetFPID().GetLibItemName().empty() ) // Handle old boards.
670 {
671 FOOTPRINT* fpCopy = static_cast<FOOTPRINT*>( footprint->Duplicate( IGNORE_PARENT_GROUP ) );
672
673 // Reset reference designator, group membership, and zone offset before saving
674
675 fpCopy->SetReference( "REF**" );
676 fpCopy->SetParentGroup( nullptr );
677
678 for( ZONE* zone : fpCopy->Zones() )
679 zone->Move( -fpCopy->GetPosition() );
680
681 adapter->SaveFootprint( nickname, fpCopy, true );
682
683 delete fpCopy;
684 }
685 }
686 catch( const IO_ERROR& ioe )
687 {
688 DisplayError( this, ioe.What() );
689 }
690
691 if( map )
692 {
693 LIB_ID id = footprint->GetFPID();
694 id.SetLibNickname( nickname );
695 footprint->SetFPID( id );
696 }
697 }
698}
699
700
702{
703 if( !aFootprint ) // Happen if no footprint loaded
704 return false;
705
706 PAD_TOOL* padTool = m_toolManager->GetTool<PAD_TOOL>();
707
708 if( padTool->InPadEditMode() )
710
711 wxString libraryName = aFootprint->GetFPID().GetLibNickname();
712 wxString footprintName = aFootprint->GetFPID().GetLibItemName();
713 bool nameChanged = m_footprintNameWhenLoaded != footprintName;
714
715 if( aFootprint->GetLink() != niluuid )
716 {
717 if( SaveFootprintToBoard( false ) )
718 {
719 m_footprintNameWhenLoaded = footprintName;
720 return true;
721 }
722
723 return false;
724 }
725 else if( libraryName.IsEmpty() || footprintName.IsEmpty() )
726 {
727 if( SaveFootprintAs( aFootprint ) )
728 {
729 m_footprintNameWhenLoaded = footprintName;
730 SyncLibraryTree( true );
731 return true;
732 }
733
734 return false;
735 }
736
738
739 // Legacy libraries are readable, but modifying legacy format is not allowed
740 // So prompt the user if he try to add/replace a footprint in a legacy lib
741 wxString libfullname;
742
743 if( std::optional<wxString> optUri = manager.GetFullURI( LIBRARY_TABLE_TYPE::FOOTPRINT, libraryName ) )
744 libfullname = *optUri;
745 else
746 return false;
747
749 {
751 return false;
752 }
753
754 if( nameChanged )
755 {
756 LIB_ID oldFPID( libraryName, m_footprintNameWhenLoaded );
757 DeleteFootprintFromLibrary( oldFPID, false );
758 }
759
760 if( !SaveFootprintInLibrary( aFootprint, libraryName ) )
761 return false;
762
763 if( nameChanged )
764 {
765 m_footprintNameWhenLoaded = footprintName;
766 SyncLibraryTree( true );
767 }
768
769 return true;
770}
771
772
774{
777
778 LIB_ID fpID = aFootprint->GetFPID();
779 wxString libraryName = fpID.GetLibNickname();
780 wxString footprintName = fpID.GetLibItemName();
781
782 // Legacy libraries are readable, but modifying legacy format is not allowed
783 // So prompt the user if he try to add/replace a footprint in a legacy lib
784 if( std::optional<wxString> optUri = manager.GetFullURI( LIBRARY_TABLE_TYPE::FOOTPRINT, libraryName ) )
785 {
787 {
789 return false;
790 }
791 }
792 else
793 {
794 return false;
795 }
796
797 int i = 1;
798 wxString newName = footprintName;
799
800 // Append a number to the name until the name is unique in the library.
801 while( adapter->FootprintExists( libraryName, newName ) )
802 newName.Printf( "%s_%d", footprintName, i++ );
803
804 aFootprint->SetFPID( LIB_ID( libraryName, newName ) );
805
806 if( aFootprint->GetValue() == footprintName )
807 aFootprint->SetValue( newName );
808
809 return SaveFootprintInLibrary( aFootprint, libraryName );
810}
811
812
814 const wxString& aLibraryName )
815{
816 try
817 {
818 aFootprint->SetFPID( LIB_ID( wxEmptyString, aFootprint->GetFPID().GetLibItemName() ) );
819
821 adapter->SaveFootprint( aLibraryName, aFootprint );
822
823 aFootprint->SetFPID( LIB_ID( aLibraryName, aFootprint->GetFPID().GetLibItemName() ) );
824
825 if( aFootprint == GetBoard()->GetFirstFootprint() )
826 setFPWatcher( aFootprint );
827
828 return true;
829 }
830 catch( const IO_ERROR& ioe )
831 {
832 DisplayError( this, ioe.What() );
833
834 aFootprint->SetFPID( LIB_ID( aLibraryName, aFootprint->GetFPID().GetLibItemName() ) );
835 return false;
836 }
837}
838
839
841{
842 // update footprint in the current board,
843 // not just add it to the board with total disregard for the netlist...
844 PCB_EDIT_FRAME* pcbframe = (PCB_EDIT_FRAME*) Kiway().Player( FRAME_PCB_EDITOR, false );
845
846 if( pcbframe == nullptr ) // happens when the board editor is not active (or closed)
847 {
848 ShowInfoBarError( _( "No board currently open." ) );
849 return false;
850 }
851
852 BOARD* mainpcb = pcbframe->GetBoard();
853 FOOTPRINT* sourceFootprint = nullptr;
854 FOOTPRINT* editorFootprint = GetBoard()->GetFirstFootprint();
855
856 if( !editorFootprint )
857 return false;
858
859 // Search the old footprint (source) if exists
860 // Because this source could be deleted when editing the main board...
861 if( editorFootprint->GetLink() != niluuid ) // this is not a new footprint ...
862 {
863 sourceFootprint = nullptr;
864
865 for( FOOTPRINT* candidate : mainpcb->Footprints() )
866 {
867 if( editorFootprint->GetLink() == candidate->m_Uuid )
868 {
869 sourceFootprint = candidate;
870 break;
871 }
872 }
873 }
874
875 if( !aAddNew && sourceFootprint == nullptr ) // source not found
876 {
877 DisplayError( this, _( "Unable to find the footprint on the main board.\nCannot save." ) );
878 return false;
879 }
880
881 TOOL_MANAGER* pcb_ToolManager = pcbframe->GetToolManager();
882
883 if( aAddNew && pcb_ToolManager->GetTool<BOARD_EDITOR_CONTROL>()->PlacingFootprint() )
884 {
885 DisplayError( this, _( "Previous footprint placement still in progress." ) );
886 return false;
887 }
888
890 BOARD_COMMIT commit( pcbframe );
891
892 // Create a copy for the board, first using Clone() to keep existing Uuids, and then either
893 // resetting the uuids to the board values or assigning new Uuids.
894 FOOTPRINT* newFootprint = static_cast<FOOTPRINT*>( editorFootprint->Clone() );
895 newFootprint->SetParent( mainpcb );
896 newFootprint->SetLink( niluuid );
897
898 auto fixUuid =
899 [&]( KIID& aUuid )
900 {
901 if( editorFootprint->GetLink() != niluuid && m_boardFootprintUuids.count( aUuid ) )
902 aUuid = m_boardFootprintUuids[ aUuid ];
903 else
904 aUuid = KIID();
905 };
906
907 fixUuid( const_cast<KIID&>( newFootprint->m_Uuid ) );
908
909 newFootprint->RunOnChildren(
910 [&]( BOARD_ITEM* aChild )
911 {
912 fixUuid( const_cast<KIID&>( aChild->m_Uuid ) );
913 },
915
916 // Right now, we only show the "Unconnected" net in the footprint editor, but this is still
917 // referenced in the footprint. So we need to update the net pointers in the footprint to
918 // point to the nets in the main board.
919 newFootprint->RunOnChildren(
920 [&]( BOARD_ITEM* aChild )
921 {
922 if( BOARD_CONNECTED_ITEM* conn = dynamic_cast<BOARD_CONNECTED_ITEM*>( aChild ) )
923 {
924 NETINFO_ITEM* net = conn->GetNet();
925 auto& netmap = mainpcb->GetNetInfo().NetsByName();
926
927 if( net )
928 {
929 auto it = netmap.find( net->GetNetname() );
930
931 if( it != netmap.end() )
932 conn->SetNet( it->second );
933 }
934
935 }
936 },
938
939 BOARD_DESIGN_SETTINGS& bds = m_pcb->GetDesignSettings();
940
943 bds.m_StyleFPBarcodes );
944
945 if( sourceFootprint ) // this is an update command
946 {
947 // In the main board the new footprint replaces the old one (pos, orient, ref, value,
948 // connections and properties are kept) and the sourceFootprint (old footprint) is
949 // deleted
950 pcbframe->ExchangeFootprint( sourceFootprint, newFootprint, commit );
951 commit.Push( _( "Update Footprint" ) );
952 }
953 else // This is an insert command
954 {
955 KIGFX::VIEW_CONTROLS* viewControls = pcbframe->GetCanvas()->GetViewControls();
956 VECTOR2D cursorPos = viewControls->GetCursorPosition();
957
958 commit.Add( newFootprint );
959 viewControls->SetCrossHairCursorPosition( VECTOR2D( 0, 0 ), false );
960 pcbframe->PlaceFootprint( newFootprint );
961 newFootprint->SetPosition( VECTOR2I( 0, 0 ) );
962 viewControls->SetCrossHairCursorPosition( cursorPos, false );
963 const_cast<KIID&>( newFootprint->m_Uuid ) = KIID();
964 commit.Push( _( "Insert Footprint" ) );
965
966 pcbframe->Raise();
967 pcb_ToolManager->RunAction( PCB_ACTIONS::placeFootprint, newFootprint );
968 }
969
970 newFootprint->ClearFlags();
971
972 return true;
973}
974
975
977{
978public:
979 SAVE_AS_DIALOG( FOOTPRINT_EDIT_FRAME* aParent, const wxString& aFootprintName,
980 const wxString& aLibraryPreselect,
981 std::function<bool( wxString libName, wxString fpName )> aValidator ) :
982 EDA_LIST_DIALOG( aParent, _( "Save Footprint As" ), false ),
983 m_validator( std::move( aValidator ) )
984 {
986 std::vector<wxString> nicknames = adapter->GetLibraryNames();
987 wxArrayString headers;
988 std::vector<wxArrayString> itemsToDisplay;
989
990 aParent->GetLibraryItemsForListDialog( headers, itemsToDisplay );
991 initDialog( headers, itemsToDisplay, aLibraryPreselect );
992
993 SetListLabel( _( "Save in library:" ) );
994 SetOKLabel( _( "Save" ) );
995
996 wxBoxSizer* bNameSizer = new wxBoxSizer( wxHORIZONTAL );
997
998 wxStaticText* label = new wxStaticText( this, wxID_ANY, _( "Name:" ) );
999 bNameSizer->Add( label, 0, wxALIGN_CENTER_VERTICAL|wxTOP|wxBOTTOM|wxLEFT, 5 );
1000
1001 m_fpNameCtrl = new wxTextCtrl( this, wxID_ANY, aFootprintName );
1002 bNameSizer->Add( m_fpNameCtrl, 1, wxALIGN_CENTER_VERTICAL|wxALL, 5 );
1003
1004 wxTextValidator nameValidator( wxFILTER_EXCLUDE_CHAR_LIST );
1005 nameValidator.SetCharExcludes( FOOTPRINT::StringLibNameInvalidChars( false ) );
1006 m_fpNameCtrl->SetValidator( nameValidator );
1007
1008 wxButton* newLibraryButton = new wxButton( this, ID_MAKE_NEW_LIBRARY, _( "New Library..." ) );
1009 m_ButtonsSizer->Prepend( 80, 20 );
1010 m_ButtonsSizer->Prepend( newLibraryButton, 0, wxALIGN_CENTER_VERTICAL|wxLEFT|wxRIGHT, 10 );
1011
1012 GetSizer()->Prepend( bNameSizer, 0, wxEXPAND|wxTOP|wxLEFT|wxRIGHT, 5 );
1013
1014 // If a footprint name was specified, disable loading of previously-saved state
1015 if( !aFootprintName.IsEmpty() )
1017
1018 Bind( wxEVT_BUTTON,
1019 [this]( wxCommandEvent& )
1020 {
1021 EndModal( ID_MAKE_NEW_LIBRARY );
1023
1024 // Move nameTextCtrl to the head of the tab-order
1025 if( GetChildren().DeleteObject( m_fpNameCtrl ) )
1026 GetChildren().Insert( m_fpNameCtrl );
1027
1029
1031
1032 Layout();
1033 GetSizer()->Fit( this );
1034
1035 Centre();
1036 }
1037
1038 wxString GetFPName()
1039 {
1040 wxString footprintName = m_fpNameCtrl->GetValue();
1041 footprintName.Trim( true );
1042 footprintName.Trim( false );
1043 return footprintName;
1044 }
1045
1046protected:
1047 bool TransferDataToWindow() override
1048 {
1049 // Respond to any filter text loaded from previously-saved state
1050 wxCommandEvent dummy;
1052
1053 return true;
1054 }
1055
1057 {
1058 return m_validator( GetTextSelection(), GetFPName() );
1059 }
1060
1061private:
1062 wxTextCtrl* m_fpNameCtrl;
1063 std::function<bool( wxString libName, wxString fpName )> m_validator;
1064};
1065
1066
1068{
1069 if( aFootprint == nullptr )
1070 return false;
1071
1072 LIBRARY_MANAGER& manager = Pgm().GetLibraryManager();
1074
1075 SetMsgPanel( aFootprint );
1076
1077 wxString libraryName = aFootprint->GetFPID().GetLibNickname();
1078 wxString footprintName = aFootprint->GetFPID().GetLibItemName();
1079 bool updateValue = aFootprint->GetValue() == footprintName;
1080 bool done = false;
1081 bool footprintExists = false;
1082
1083 while( !done )
1084 {
1085 SAVE_AS_DIALOG dlg( this, footprintName, libraryName,
1086 [&]( const wxString& newLib, const wxString& newName )
1087 {
1088 if( newLib.IsEmpty() )
1089 {
1090 wxMessageBox( _( "A library must be specified." ) );
1091 return false;
1092 }
1093
1094 if( newName.IsEmpty() )
1095 {
1096 wxMessageBox( _( "Footprint must have a name." ) );
1097 return false;
1098 }
1099
1100 // Legacy libraries are readable, but modifying legacy format is not allowed
1101 // So prompt the user if he try to add/replace a footprint in a legacy lib
1102 if( std::optional<wxString> optUri = manager.GetFullURI( LIBRARY_TABLE_TYPE::FOOTPRINT, newLib ) )
1103 {
1105 {
1107 return false;
1108 }
1109 }
1110 else
1111 {
1112 return false;
1113 }
1114
1115 footprintExists = adapter->FootprintExists( newLib, newName );
1116
1117 if( footprintExists )
1118 {
1119 wxString msg = wxString::Format( _( "Footprint %s already exists in %s." ),
1120 newName,
1121 newLib );
1122
1123 KIDIALOG errorDlg( this, msg, _( "Confirmation" ), wxOK | wxCANCEL | wxICON_WARNING );
1124 errorDlg.SetOKLabel( _( "Overwrite" ) );
1125
1126 return errorDlg.ShowModal() == wxID_OK;
1127 }
1128
1129 return true;
1130 } );
1131
1132 int ret = dlg.ShowModal();
1133
1134 if( ret == wxID_CANCEL )
1135 {
1136 return false;
1137 }
1138 else if( ret == wxID_OK )
1139 {
1140 footprintName = dlg.GetFPName();
1141 libraryName = dlg.GetTextSelection();
1142 done = true;
1143 }
1144 else if( ret == ID_MAKE_NEW_LIBRARY )
1145 {
1146 wxFileName fn = CreateNewLibrary( _( "New Footprint Library" ),
1147 Prj().GetRString( PROJECT::PCB_LIB_PATH ) );
1148
1149 Prj().SetRString( PROJECT::PCB_LIB_PATH, fn.GetPath() );
1150 Prj().SetRString( PROJECT::PCB_LIB_NICKNAME, fn.GetName() );
1151 libraryName = fn.GetName();
1152 }
1153 }
1154
1155 aFootprint->SetFPID( LIB_ID( libraryName, footprintName ) );
1156
1157 if( updateValue )
1158 aFootprint->SetValue( footprintName );
1159
1160 if( !SaveFootprintInLibrary( aFootprint, libraryName ) )
1161 return false;
1162
1163 // Once saved-as a board footprint is no longer a board footprint
1164 aFootprint->SetLink( niluuid );
1165
1166 wxString fmt = footprintExists ? _( "Footprint '%s' replaced in '%s'" )
1167 : _( "Footprint '%s' added to '%s'" );
1168
1169 wxString msg = wxString::Format( fmt, footprintName.GetData(), libraryName.GetData() );
1170 SetStatusText( msg );
1171 UpdateTitle();
1173
1174 return true;
1175}
1176
1177
1179{
1181 {
1182 wxString msg = wxString::Format( _( "Revert '%s' to last version saved?" ),
1183 GetLoadedFPID().GetLibItemName().wx_str() );
1184
1185 if( ConfirmRevertDialog( this, msg ) )
1186 {
1187 Clear_Pcb( false );
1188 AddFootprintToBoard( static_cast<FOOTPRINT*>( m_originalFootprintCopy->Clone() ) );
1189
1190 Zoom_Automatique( false );
1191
1192 Update3DView( true, true );
1193
1195 GetScreen()->SetContentModified( false );
1196
1197 UpdateView();
1198 GetCanvas()->Refresh();
1199
1200 return true;
1201 }
1202 }
1203
1204 return false;
1205}
1206
1207
1208FOOTPRINT* PCB_BASE_FRAME::CreateNewFootprint( wxString aFootprintName, const wxString& aLibName )
1209{
1210 if( aFootprintName.IsEmpty() )
1211 aFootprintName = _( "Untitled" );
1212
1213 int footprintAttrs = FP_SMD;
1214
1215 if( !aLibName.IsEmpty() )
1216 {
1218 std::vector<wxString> fpnames;
1219 wxString baseName = aFootprintName;
1220 int idx = 1;
1221
1222 // Make sure the name is unique
1223 while( adapter->FootprintExists( aLibName, aFootprintName ) )
1224 aFootprintName = baseName + wxString::Format( wxS( "_%d" ), idx++ );
1225
1226 // Try to infer the footprint attributes from an existing footprint in the library
1227 try
1228 {
1229 fpnames = adapter->GetFootprintNames( aLibName, true );
1230
1231 if( !fpnames.empty() )
1232 {
1233 std::unique_ptr<FOOTPRINT> fp( adapter->LoadFootprint( aLibName, fpnames.back(), false ) );
1234
1235 if( fp )
1236 footprintAttrs = fp->GetAttributes();
1237 }
1238 }
1239 catch( ... )
1240 {
1241 // best efforts
1242 }
1243 }
1244
1245 // Create the new footprint and add it to the head of the linked list of footprints
1246 FOOTPRINT* footprint = new FOOTPRINT( GetBoard() );
1247
1248 // Update its name in lib
1249 footprint->SetFPID( LIB_ID( wxEmptyString, aFootprintName ) );
1250
1251 footprint->SetAttributes( footprintAttrs );
1252
1253 PCB_LAYER_ID txt_layer;
1254 VECTOR2I default_pos;
1256
1257 if( settings.m_DefaultFPTextItems.size() > 0 )
1258 {
1259 footprint->Reference().SetText( settings.m_DefaultFPTextItems[0].m_Text );
1260 footprint->Reference().SetVisible( settings.m_DefaultFPTextItems[0].m_Visible );
1261 }
1262
1263 txt_layer = settings.m_DefaultFPTextItems[0].m_Layer;
1264 footprint->Reference().SetLayer( txt_layer );
1265 default_pos.y -= settings.GetTextSize( txt_layer ).y / 2;
1266 footprint->Reference().SetPosition( default_pos );
1267 default_pos.y += settings.GetTextSize( txt_layer ).y;
1268
1269 if( settings.m_DefaultFPTextItems.size() > 1 )
1270 {
1271 footprint->Value().SetText( settings.m_DefaultFPTextItems[1].m_Text );
1272 footprint->Value().SetVisible( settings.m_DefaultFPTextItems[1].m_Visible );
1273 }
1274
1275 txt_layer = settings.m_DefaultFPTextItems[1].m_Layer;
1276 footprint->Value().SetLayer( txt_layer );
1277 default_pos.y += settings.GetTextSize( txt_layer ).y / 2;
1278 footprint->Value().SetPosition( default_pos );
1279 default_pos.y += settings.GetTextSize( txt_layer ).y;
1280
1281 for( size_t i = 2; i < settings.m_DefaultFPTextItems.size(); ++i )
1282 {
1283 PCB_TEXT* textItem = new PCB_TEXT( footprint );
1284 textItem->SetText( settings.m_DefaultFPTextItems[i].m_Text );
1285 txt_layer = (PCB_LAYER_ID) settings.m_DefaultFPTextItems[i].m_Layer;
1286 textItem->SetLayer( txt_layer );
1287 default_pos.y += settings.GetTextSize( txt_layer ).y / 2;
1288 textItem->SetPosition( default_pos );
1289 default_pos.y += settings.GetTextSize( txt_layer ).y;
1290 footprint->GraphicalItems().push_back( textItem );
1291 }
1292
1293 if( footprint->GetReference().IsEmpty() )
1294 footprint->SetReference( aFootprintName );
1295
1296 if( footprint->GetValue().IsEmpty() )
1297 footprint->SetValue( aFootprintName );
1298
1299 footprint->RunOnChildren(
1300 [&]( BOARD_ITEM* aChild )
1301 {
1302 if( aChild->Type() == PCB_FIELD_T || aChild->Type() == PCB_TEXT_T )
1303 {
1304 PCB_TEXT* textItem = static_cast<PCB_TEXT*>( aChild );
1305 PCB_LAYER_ID layer = textItem->GetLayer();
1306
1307 textItem->SetTextThickness( settings.GetTextThickness( layer ) );
1308 textItem->SetTextSize( settings.GetTextSize( layer ) );
1309 textItem->SetItalic( settings.GetTextItalic( layer ) );
1310 textItem->SetKeepUpright( settings.GetTextUpright( layer ) );
1311 }
1312 },
1314
1315 SetMsgPanel( footprint );
1316 return footprint;
1317}
1318
1319
1321 std::vector<wxArrayString>& aItemsToDisplay )
1322{
1323 aHeaders.Add( _( "Library" ) );
1324 aHeaders.Add( _( "Description" ) );
1325
1329 std::vector<wxString> nicknames = adapter->GetLibraryNames();
1330 std::vector<wxArrayString> unpinned;
1331
1332 for( const wxString& nickname : nicknames )
1333 {
1334 wxArrayString item;
1335 wxString description = adapter->GetLibraryDescription( nickname ).value_or( wxEmptyString );
1336
1337 if( alg::contains( project.m_PinnedFootprintLibs, nickname )
1338 || alg::contains( cfg->m_Session.pinned_fp_libs, nickname ) )
1339 {
1340 item.Add( LIB_TREE_MODEL_ADAPTER::GetPinningSymbol() + nickname );
1341 item.Add( description );
1342 aItemsToDisplay.push_back( item );
1343 }
1344 else
1345 {
1346 item.Add( nickname );
1347 item.Add( description );
1348 unpinned.push_back( item );
1349 }
1350 }
1351
1352 std::ranges::copy( unpinned, std::back_inserter( aItemsToDisplay ) );
1353}
static TOOL_ACTION selectionClear
Clear the current selection.
Definition actions.h:224
void SetContentModified(bool aModified=true)
Definition base_screen.h:59
virtual void Push(const wxString &aMessage=wxEmptyString, int aCommitFlags=0) override
Execute the changes.
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
Container for design settings for a BOARD object.
std::vector< TEXT_ITEM_INFO > m_DefaultFPTextItems
bool GetTextUpright(PCB_LAYER_ID aLayer) const
int GetTextThickness(PCB_LAYER_ID aLayer) const
Return the default text thickness from the layer class for the given layer.
bool GetTextItalic(PCB_LAYER_ID aLayer) const
VECTOR2I GetTextSize(PCB_LAYER_ID aLayer) const
Return the default text size from the layer class for the given layer.
Handle actions specific to the board editor in PcbNew.
bool PlacingFootprint() const
Re-entrancy checker for above.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition board_item.h:83
virtual PCB_LAYER_ID GetLayer() const
Return the primary layer this item is on.
Definition board_item.h:236
virtual void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
Definition board_item.h:284
Information pertinent to a Pcbnew printed circuit board.
Definition board.h:322
const NETINFO_LIST & GetNetInfo() const
Definition board.h:996
void BuildListOfNets()
Definition board.h:959
FOOTPRINT * GetFirstFootprint() const
Get the first footprint on the board or nullptr.
Definition board.h:530
const FOOTPRINTS & Footprints() const
Definition board.h:363
COMMIT & Add(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Add a new item to the model.
Definition commit.h:78
void OptOut(wxWindow *aWindow)
Opt out of control state saving.
void SetInitialFocus(wxWindow *aWindow)
Sets the window (usually a wxTextCtrl) that should be focused when the dialog is shown.
Definition dialog_shim.h:82
void SetupStandardButtons(std::map< int, wxString > aLabels={})
int ShowModal() override
virtual void ClearUndoRedoList()
Clear the undo and redo list using ClearUndoORRedoList()
void ShowInfoBarError(const wxString &aErrorMsg, bool aShowCloseButton=false, WX_INFOBAR::MESSAGE_TYPE aType=WX_INFOBAR::MESSAGE_TYPE::GENERIC)
Show the WX_INFOBAR displayed on the top of the canvas with a message and an error icon on the left o...
void SetMsgPanel(const std::vector< MSG_PANEL_ITEM > &aList)
Clear the message panel and populates it with the contents of aList.
virtual void ReCreateHToolbar()
virtual void Zoom_Automatique(bool aWarpPointer)
Redraw the screen with best zoom level and the best centering that shows all the page or the board.
bool LibraryFileBrowser(const wxString &aTitle, bool doOpen, wxFileName &aFilename, const wxString &wildcard, const wxString &ext, bool isDirectory, FILEDLG_HOOK_NEW_LIBRARY *aFileDlgHook=nullptr)
KIGFX::VIEW_CONTROLS * GetViewControls() const
Return a pointer to the #VIEW_CONTROLS instance used in the panel.
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
const KIID m_Uuid
Definition eda_item.h:521
KICAD_T Type() const
Returns the type of object.
Definition eda_item.h:110
void ClearFlags(EDA_ITEM_FLAGS aMask=EDA_ITEM_ALL_FLAGS)
Definition eda_item.h:149
virtual void SetParentGroup(EDA_GROUP *aGroup)
Definition eda_item.h:115
virtual void SetParent(EDA_ITEM *aParent)
Definition eda_item.h:113
A dialog which shows:
void textChangeInFilterBox(wxCommandEvent &event) override
void SetOKLabel(const wxString &aLabel)
void initDialog(const wxArrayString &aItemHeaders, const std::vector< wxArrayString > &aItemList, const wxString &aPreselectText)
wxString GetTextSelection(int aColumn=0)
Return the selected text from aColumn in the wxListCtrl in the dialog.
void SetListLabel(const wxString &aLabel)
void AddExtraCheckbox(const wxString &aLabel, bool *aValuePtr)
Add a checkbox value to the dialog.
void GetExtraCheckboxValues()
Fills in the value pointers from the checkboxes after the dialog has run.
EDA_LIST_DIALOG(wxWindow *aParent, const wxString &aTitle, const wxArrayString &aItemHeaders, const std::vector< wxArrayString > &aItemList, const wxString &aPreselectText=wxEmptyString, bool aSortList=true)
void SetTextSize(VECTOR2I aNewSize, bool aEnforceMinTextSize=true)
Definition eda_text.cpp:544
virtual void SetVisible(bool aVisible)
Definition eda_text.cpp:398
void SetTextThickness(int aWidth)
The TextThickness is that set by the user.
Definition eda_text.cpp:296
void SetKeepUpright(bool aKeepUpright)
Definition eda_text.cpp:437
virtual void SetText(const wxString &aText)
Definition eda_text.cpp:282
void SetItalic(bool aItalic)
Set the text to be italic - this will also update the font if needed.
Definition eda_text.cpp:319
void SyncLibraryTree(bool aProgress)
Synchronize the footprint library tree to the current state of the footprint library table.
bool SaveFootprintInLibrary(FOOTPRINT *aFootprint, const wxString &aLibraryName)
bool SaveFootprintAs(FOOTPRINT *aFootprint)
bool DuplicateFootprint(FOOTPRINT *aFootprint)
void ExportFootprint(FOOTPRINT *aFootprint)
Create a file containing only one footprint.
LIB_ID GetLoadedFPID() const
Return the LIB_ID of the part being edited.
bool SaveFootprintToBoard(bool aAddNew)
bool SaveFootprint(FOOTPRINT *aFootprint)
Save in an existing library a given footprint.
FOOTPRINT * ImportFootprint(const wxString &aName=wxT(""))
Read a file containing only one footprint.
std::map< KIID, KIID > m_boardFootprintUuids
bool IsContentModified() const override
Get if any footprints or libraries have been modified but not saved.
bool Clear_Pcb(bool doAskAboutUnsavedChanges)
Delete all and reinitialize the current board.
Definition initpcb.cpp:108
void AddFootprintToBoard(FOOTPRINT *aFootprint) override
Override from PCB_BASE_EDIT_FRAME which adds a footprint to the editor's dummy board,...
bool DeleteFootprintFromLibrary(const LIB_ID &aFPID, bool aConfirm)
Delete the given footprint from its library.
std::unique_ptr< FOOTPRINT > m_originalFootprintCopy
FOOTPRINT_EDITOR_SETTINGS * GetSettings()
An interface to the global shared library manager that is schematic-specific and linked to one projec...
bool IsFootprintLibWritable(const wxString &aNickname)
Return true if the library given by aNickname is writable.
void DeleteFootprint(const wxString &aNickname, const wxString &aFootprintName)
Deletes the aFootprintName from the library given by aNickname.
SAVE_T SaveFootprint(const wxString &aNickname, const FOOTPRINT *aFootprint, bool aOverwrite=true)
Write aFootprint to an existing library given by aNickname.
std::vector< wxString > GetFootprintNames(const wxString &aNickname, bool aBestEfforts=false)
Retrieves a list of footprint names contained in a given loaded library.
FOOTPRINT * LoadFootprint(const wxString &aNickname, const wxString &aName, bool aKeepUUID)
Load a FOOTPRINT having aName from the library given by aNickname.
std::optional< LIB_STATUS > LoadOne(LIB_DATA *aLib) override
Loads or reloads the given library, if it exists.
bool FootprintExists(const wxString &aNickname, const wxString &aName)
Component library viewer main window.
void SetPosition(const VECTOR2I &aPos) override
void SetFPID(const LIB_ID &aFPID)
Definition footprint.h:352
void SetLink(const KIID &aLink)
Definition footprint.h:1076
ZONES & Zones()
Definition footprint.h:312
void RunOnChildren(const std::function< void(BOARD_ITEM *)> &aFunction, RECURSE_MODE aMode) const override
Invoke a function on all children.
void SetAttributes(int aAttributes)
Definition footprint.h:418
EDA_ITEM * Clone() const override
Invoke a function on all children.
PCB_FIELD & Value()
read/write accessors:
Definition footprint.h:787
const LIB_ID & GetFPID() const
Definition footprint.h:351
void SetReference(const wxString &aReference)
Definition footprint.h:757
void ApplyDefaultSettings(const BOARD &board, bool aStyleFields, bool aStyleText, bool aStyleShapes, bool aStyleDimensions, bool aStyleBarcodes)
Apply default board settings to the footprint field text properties.
void SetValue(const wxString &aValue)
Definition footprint.h:778
PCB_FIELD & Reference()
Definition footprint.h:788
void Add(BOARD_ITEM *aItem, ADD_MODE aMode=ADD_MODE::INSERT, bool aSkipConnectivity=false) override
Removes an item from the container.
KIID GetLink() const
Definition footprint.h:1075
static const wxChar * StringLibNameInvalidChars(bool aUserReadable)
Test for validity of the name in a library of the footprint ( no spaces, dir separators ....
const wxString & GetValue() const
Definition footprint.h:773
const wxString & GetReference() const
Definition footprint.h:751
VECTOR2I GetPosition() const override
Definition footprint.h:327
DRAWINGS & GraphicalItems()
Definition footprint.h:309
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
virtual const wxString What() const
A composite of Problem() and Where()
Helper class to create more flexible dialogs, including 'do not show again' checkbox handling.
Definition kidialog.h:42
void DoNotShowCheckbox(wxString file, int line)
Shows the 'do not show again' checkbox.
Definition kidialog.cpp:55
int ShowModal() override
Definition kidialog.cpp:93
An interface for classes handling user events controlling the view behavior such as zooming,...
virtual void SetCrossHairCursorPosition(const VECTOR2D &aPosition, bool aWarpView=true)=0
Move the graphic crosshair cursor to the requested position expressed in world coordinates.
VECTOR2D GetCursorPosition() const
Return the current cursor position in world coordinates.
Definition kiid.h:49
PROJECT & Prj() const
Return a reference to the PROJECT associated with this KIWAY.
virtual KIWAY_PLAYER * Player(FRAME_T aFrameType, bool doCreate=true, wxTopLevelWindow *aParent=nullptr)
Return the KIWAY_PLAYER* given a FRAME_T.
Definition kiway.cpp:403
virtual PROJECT & Prj() const
Return the PROJECT associated with this KIWAY.
Definition kiway.cpp:200
std::optional< wxString > GetLibraryDescription(const wxString &aNickname) const
std::vector< wxString > GetLibraryNames() const
Returns a list of library nicknames that are available (skips any that failed to load)
void ReloadTables(LIBRARY_TABLE_SCOPE aScope, std::initializer_list< LIBRARY_TABLE_TYPE > aTablesToLoad={})
std::optional< LIBRARY_TABLE * > Table(LIBRARY_TABLE_TYPE aType, LIBRARY_TABLE_SCOPE aScope)
Retrieves a given table; creating a new empty project table if a valid project is loaded and the give...
std::optional< wxString > GetFullURI(LIBRARY_TABLE_TYPE aType, const wxString &aNickname, bool aSubstituted=false) const
Return the full location specifying URI for the LIB, either in original UI form or in environment var...
void SetNickname(const wxString &aNickname)
void SetType(const wxString &aType)
void SetURI(const wxString &aUri)
A logical library item identifier and consists of various portions much like a URI.
Definition lib_id.h:49
bool IsValid() const
Check if this LID_ID is valid.
Definition lib_id.h:172
int SetLibNickname(const UTF8 &aLibNickname)
Override the logical library name portion of the LIB_ID to aLibNickname.
Definition lib_id.cpp:100
const UTF8 & GetLibItemName() const
Definition lib_id.h:102
const UTF8 & GetLibNickname() const
Return the logical library name portion of a LIB_ID.
Definition lib_id.h:87
static const wxString GetPinningSymbol()
Handle the data for a net.
Definition netinfo.h:54
const wxString & GetNetname() const
Definition netinfo.h:112
const NETNAMES_MAP & NetsByName() const
Return the name map, at least for python.
Definition netinfo.h:238
bool InPadEditMode()
Definition pad_tool.h:63
static TOOL_ACTION recombinePad
static TOOL_ACTION placeFootprint
wxString CreateNewLibrary(const wxString &aDialogTitle, const wxString &aInitialPath=wxEmptyString)
If a library name is given, creates a new footprint library in the project folder with the given name...
wxString SelectLibrary(const wxString &aDialogTitle, const wxString &aListLabel, const std::vector< std::pair< wxString, bool * > > &aExtraCheckboxes={})
Put up a dialog and allows the user to pick a library, for unspecified use.
wxString createNewLibrary(const wxString &aDialogTitle, const wxString &aLibName, const wxString &aInitialPath, std::optional< LIBRARY_TABLE_SCOPE > aScope=std::nullopt)
Create a new library in the given table.
wxString CreateNewProjectLibrary(const wxString &aDialogTitle, const wxString &aLibName)
bool AddLibrary(const wxString &aDialogTitle, const wxString &aLibName=wxEmptyString, std::optional< LIBRARY_TABLE_SCOPE > aScope=std::nullopt)
Add an existing library to either the global or project library table.
void setFPWatcher(FOOTPRINT *aFootprint)
Create or removes a watcher on the specified footprint.
FOOTPRINT * CreateNewFootprint(wxString aFootprintName, const wxString &aLibName)
Create a new footprint at position 0,0.
void GetLibraryItemsForListDialog(wxArrayString &aHeaders, std::vector< wxArrayString > &aItemsToDisplay)
PCB_DRAW_PANEL_GAL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
PCB_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
BOARD * GetBoard() const
virtual BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Return the BOARD_DESIGN_SETTINGS for the open project.
void PlaceFootprint(FOOTPRINT *aFootprint, bool aRecreateRatsnest=true, std::optional< VECTOR2I > aPosition=std::nullopt)
Place aFootprint at the current cursor position (or provided one) and updates footprint coordinates w...
virtual void Update3DView(bool aMarkDirty, bool aRefresh, const wxString *aTitle=nullptr)
Update the 3D view, if the viewer is opened by this frame.
The main frame for Pcbnew.
void ExportFootprintsToLibrary(bool aStoreInNewLib, const wxString &aLibName=wxEmptyString, wxString *aLibPath=nullptr)
Save footprints in a library:
void ExchangeFootprint(FOOTPRINT *aExisting, FOOTPRINT *aNew, BOARD_COMMIT &aCommit, bool deleteExtraTexts=true, bool resetTextLayers=true, bool resetTextEffects=true, bool resetTextPositions=true, bool resetTextContent=true, bool resetFabricationAttrs=true, bool resetClearanceOverrides=true, bool reset3DModels=true, bool *aUpdated=nullptr)
Replace aExisting footprint by aNew footprint using the Existing footprint settings (position,...
A #PLUGIN derivation for saving and loading Pcbnew s-expression formatted files.
void Format(const BOARD_ITEM *aItem) const
Output aItem to aFormatter in s-expression format.
std::string GetStringOutput(bool doClear)
static PLUGIN_REGISTRY * Instance()
Definition pcb_io_mgr.h:95
PCB_FILE_T
The set of file types that the PCB_IO_MGR knows about, and for which there has been a plugin written,...
Definition pcb_io_mgr.h:56
@ KICAD_SEXP
S-expression Pcbnew file format.
Definition pcb_io_mgr.h:58
@ LEGACY
Legacy Pcbnew file formats prior to s-expression.
Definition pcb_io_mgr.h:59
static PCB_IO * FindPlugin(PCB_FILE_T aFileType)
Return a #PLUGIN which the caller can use to import, export, save, or load design documents.
static PCB_FILE_T GuessPluginTypeFromLibPath(const wxString &aLibPath, int aCtl=0)
Return a plugin type given a footprint library's libPath.
static const wxString ShowType(PCB_FILE_T aFileType)
Return a brief name for a plugin given aFileType enum.
virtual void SetPosition(const VECTOR2I &aPos) override
Definition pcb_text.h:84
virtual COMMON_SETTINGS * GetCommonSettings() const
Definition pgm_base.cpp:535
virtual LIBRARY_MANAGER & GetLibraryManager() const
Definition pgm_base.h:134
The backing store for a PROJECT, in JSON format.
static FOOTPRINT_LIBRARY_ADAPTER * FootprintLibAdapter(PROJECT *aProject)
Container for project specific data.
Definition project.h:65
@ PCB_LIB_PATH
Definition project.h:227
@ PCB_LIB_NICKNAME
Definition project.h:228
virtual const wxString GetProjectPath() const
Return the full path of the project.
Definition project.cpp:168
virtual PROJECT_FILE & GetProjectFile() const
Definition project.h:203
virtual void SetRString(RSTRING_T aStringId, const wxString &aString)
Store a "retained string", which is any session and project specific string identified in enum RSTRIN...
Definition project.cpp:340
virtual const wxString & GetRString(RSTRING_T aStringId)
Return a "retained string", which is any session and project specific string identified in enum RSTRI...
Definition project.cpp:351
bool TransferDataFromWindow() override
SAVE_AS_DIALOG(FOOTPRINT_EDIT_FRAME *aParent, const wxString &aFootprintName, const wxString &aLibraryPreselect, std::function< bool(wxString libName, wxString fpName)> aValidator)
bool TransferDataToWindow() override
std::function< bool(wxString libName, wxString fpName)> m_validator
TOOL_MANAGER * m_toolManager
TOOL_MANAGER * GetToolManager() const
Return the MVC controller.
Master controller class:
bool RunAction(const std::string &aActionName, T aParam)
Run the specified action immediately, pausing the current action to run the new one.
Handle a list of polygons defining a copper zone.
Definition zone.h:74
wxString EnsureFileExtension(const wxString &aFilename, const wxString &aExtension)
It's annoying to throw up nag dialogs when the extension isn't right.
Definition common.cpp:629
bool IsOK(wxWindow *aParent, const wxString &aMessage)
Display a yes/no dialog with aMessage and returns the user response.
Definition confirm.cpp:259
void DisplayInfoMessage(wxWindow *aParent, const wxString &aMessage, const wxString &aExtraInfo)
Display an informational message box with aMessage.
Definition confirm.cpp:230
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition confirm.cpp:202
bool ConfirmRevertDialog(wxWindow *parent, const wxString &aMessage)
Display a confirmation dialog for a revert action.
Definition confirm.cpp:118
void DisplayError(wxWindow *aParent, const wxString &aText)
Display an error or warning message box with aMessage.
Definition confirm.cpp:177
This file is part of the common library.
#define _(s)
@ RECURSE
Definition eda_item.h:51
#define IGNORE_PARENT_GROUP
Definition eda_item.h:55
wxString NormalizePath(const wxFileName &aFilePath, const ENV_VAR_MAP *aEnvVars, const wxString &aProjectPath)
Normalize a file path to an environmental variable, if possible.
Definition env_paths.cpp:73
Helper functions to substitute paths with environmental variables.
@ FP_SMD
Definition footprint.h:85
static const wxString INFO_LEGACY_LIB_WARN_DELETE(_("Modifying legacy libraries (.mod files) is not allowed\n" "Please save the current library under the new .pretty format\n" "and update your footprint lib table\n" "before deleting a footprint"))
static const wxString INFO_LEGACY_LIB_WARN_EDIT(_("Writing/modifying legacy libraries (.mod files) is not allowed\n" "Please save the current library to the new .pretty format\n" "and update your footprint lib table\n" "to save your footprint (a .kicad_mod file) in the .pretty library folder"))
static constexpr int ID_MAKE_NEW_LIBRARY
@ FRAME_PCB_EDITOR
Definition frame_type.h:42
@ FRAME_FOOTPRINT_VIEWER
Definition frame_type.h:45
@ FRAME_FOOTPRINT_EDITOR
Definition frame_type.h:43
static const std::string KiCadFootprintLibPathExtension
static const std::string KiCadFootprintFileExtension
static wxString KiCadFootprintLibFileWildcard()
static wxString KiCadFootprintLibPathWildcard()
std::unique_ptr< T > IO_RELEASER
Helper to hold and release an IO_BASE object when exceptions are thrown.
Definition io_mgr.h:33
PROJECT & Prj()
Definition kicad.cpp:637
KIID niluuid(0)
PCB_LAYER_ID
A quick note on layer IDs:
Definition layer_ids.h:60
This file contains miscellaneous commonly used macros and functions.
void Prettify(std::string &aSource, FORMAT_MODE aMode)
Pretty-prints s-expression text according to KiCad format rules.
void AllowNetworkFileSystems(wxDialog *aDialog)
Configure a file dialog to show network and virtual file systems.
Definition wxgtk/ui.cpp:717
bool contains(const _Container &__container, _Value __value)
Returns true if the container contains the given value.
Definition kicad_algo.h:100
STL namespace.
Class to handle a set of BOARD_ITEMs.
#define CTL_FOR_LIBRARY
Format output for a footprint library instead of clipboard or BOARD.
PGM_BASE & Pgm()
The global program "get" accessor.
see class PGM_BASE
KIWAY Kiway(KFCTL_STANDALONE)
std::vector< FAB_LAYER_COLOR > dummy
MODEL3D_FORMAT_TYPE fileType(const char *aFileName)
std::vector< wxString > pinned_fp_libs
Container that describes file type info.
Definition io_base.h:43
std::vector< std::string > m_FileExtensions
Filter used for file pickers if m_IsFile is true.
Definition io_base.h:47
wxString FileFilter() const
Definition io_base.cpp:40
wxString message
@ ID_MAKE_NEW_LIBRARY
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
Definition typeinfo.h:92
@ PCB_FIELD_T
class PCB_FIELD, text associated with a footprint property
Definition typeinfo.h:90
Custom text control validator definitions.
VECTOR2< int32_t > VECTOR2I
Definition vector2d.h:695
VECTOR2< double > VECTOR2D
Definition vector2d.h:694
wxString formatWildcardExt(const wxString &aWildcard)
Format wildcard extension to support case sensitive file dialogs.
Definition of file extensions used in Kicad.