KiCad PCB EDA Suite
Loading...
Searching...
No Matches
panel_fp_lib_table.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <[email protected]>
5 * Copyright (C) 2013-2021 CERN
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26
27#include <set>
28#include <wx/regex.h>
29#include <wx/dirdlg.h>
30#include <wx/filedlg.h>
31#include <wx/msgdlg.h>
32#include <functional>
33
34#include <project.h>
35#include <env_vars.h>
37#include <panel_fp_lib_table.h>
38#include <lib_id.h>
40#include <lib_table_lexer.h>
41#include <invoke_pcb_dialog.h>
42#include <bitmaps.h>
44#include <widgets/wx_grid.h>
49#include <confirm.h>
51#include <kiplatform/ui.h>
53#include <pgm_base.h>
54#include <pcb_edit_frame.h>
55#include <env_paths.h>
59#include <kiway.h>
60#include <kiway_mail.h>
61#include <pcbnew_id.h> // For ID_PCBNEW_END_LIST
63#include <paths.h>
64#include <macros.h>
65#include <project_pcb.h>
66#include <common.h>
70
71
76{
77public:
78 FP_LIB_TABLE_GRID_DATA_MODEL( DIALOG_SHIM* aParent, WX_GRID* aGrid, const LIBRARY_TABLE& aTableToEdit,
79 FOOTPRINT_LIBRARY_ADAPTER* aAdapter, const wxArrayString& aPluginChoices,
80 wxString* aMRUDirectory, const wxString& aProjectPath,
81 const std::map<PCB_IO_MGR::PCB_FILE_T, IO_BASE::IO_FILE_DESC>& aSupportedFiles ) :
82 LIB_TABLE_GRID_DATA_MODEL( aParent, aGrid, aTableToEdit, aAdapter, aPluginChoices, aMRUDirectory,
83 aProjectPath ),
84 m_supportedFpFiles( aSupportedFiles )
85 {
86 }
87
88 void SetValue( int aRow, int aCol, const wxString &aValue ) override
89 {
90 wxCHECK( aRow < (int) size(), /* void */ );
91
92 LIB_TABLE_GRID_DATA_MODEL::SetValue( aRow, aCol, aValue );
93
94 // If setting a filepath, attempt to auto-detect the format
95 if( aCol == COL_URI )
96 {
97 LIBRARY_TABLE_ROW& row = at( (size_t) aRow );
98 wxString uri = LIBRARY_MANAGER::ExpandURI( row.URI(), Pgm().GetSettingsManager().Prj() );
100
101 if( pluginType != PCB_IO_MGR::FILE_TYPE_NONE )
102 SetValue( aRow, COL_TYPE, PCB_IO_MGR::ShowType( pluginType ) );
103 }
104 }
105
106protected:
107 wxString getFileTypes( WX_GRID* aGrid, int aRow ) override
108 {
109 FP_LIB_TABLE_GRID_DATA_MODEL* table = static_cast<FP_LIB_TABLE_GRID_DATA_MODEL*>( aGrid->GetTable() );
110 LIBRARY_TABLE_ROW& tableRow = table->at( aRow );
111
112 if( tableRow.Type() == LIBRARY_TABLE_ROW::TABLE_TYPE_NAME )
113 {
114 wxString filter = _( "Footprint Library Tables" );
115#ifndef __WXOSX__
116 filter << wxString::Format( _( " (%s)|%s" ), FILEEXT::FootprintLibraryTableFileName,
118#else
119 filter << wxString::Format( _( " (%s)|%s" ), wxFileSelectorDefaultWildcardStr,
120 wxFileSelectorDefaultWildcardStr );
121#endif
122 return filter;
123 }
124
126
128 return wxEmptyString;
129
130 const IO_BASE::IO_FILE_DESC& pluginDesc = m_supportedFpFiles.at( fileType );
131
132 if( pluginDesc.m_IsFile )
133 return pluginDesc.FileFilter();
134
135 return wxEmptyString;
136 }
137
138private:
139 const std::map<PCB_IO_MGR::PCB_FILE_T, IO_BASE::IO_FILE_DESC>& m_supportedFpFiles;
140};
141
142
143
145{
146public:
147 FP_GRID_TRICKS( PANEL_FP_LIB_TABLE* aPanel, WX_GRID* aGrid, std::function<void( wxCommandEvent& )> aAddHandler ) :
148 LIB_TABLE_GRID_TRICKS( aGrid, aAddHandler ),
149 m_panel( aPanel )
150 {
152 }
153
155 {
156 return false;
157 }
158
159protected:
160 void optionsEditor( int aRow ) override
161 {
162 LIB_TABLE_GRID_DATA_MODEL* tbl = static_cast<LIB_TABLE_GRID_DATA_MODEL*>( m_grid->GetTable() );
163
164 if( tbl->GetNumberRows() > aRow )
165 {
166 LIBRARY_TABLE_ROW& row = tbl->At( static_cast<size_t>( aRow ) );
167 const wxString& options = row.Options();
168 wxString result = options;
169 std::map<std::string, UTF8> choices;
170
173 pi->GetLibraryOptions( &choices );
174
175 DIALOG_PLUGIN_OPTIONS dlg( wxGetTopLevelParent( m_grid ), row.Nickname(), choices, options, &result );
176 dlg.ShowModal();
177
178 if( options != result )
179 {
180 row.SetOptions( result );
181 m_grid->Refresh();
182 }
183 }
184 }
185
186 void openTable( const LIBRARY_TABLE_ROW& aRow ) override
187 {
188 wxFileName fn( LIBRARY_MANAGER::ExpandURI( aRow.URI(), Pgm().GetSettingsManager().Prj() ) );
189 std::shared_ptr<LIBRARY_TABLE> child = std::make_shared<LIBRARY_TABLE>( fn, LIBRARY_TABLE_SCOPE::GLOBAL, LIBRARY_TABLE_TYPE::FOOTPRINT );
190
192
193 m_panel->OpenTable( child, aRow.Nickname() );
194 }
195
196 wxString getTablePreamble() override
197 {
198 return wxT( "(fp_lib_table" );
199 }
200
205
206protected:
208};
209
210
211void PANEL_FP_LIB_TABLE::OpenTable( const std::shared_ptr<LIBRARY_TABLE>& aTable, const wxString& aTitle )
212{
213 wxString tabTitle = aTitle;
214
215 if( aTable->IsReadOnly() )
216 tabTitle += wxS( " " ) + _( "(read-only)" );
217
218 for( int ii = 2; ii < (int) m_notebook->GetPageCount(); ++ii )
219 {
220 if( m_notebook->GetPageText( ii ) == tabTitle )
221 {
222 // Something is pretty fishy with wxAuiNotebook::ChangeSelection(); on Mac at least it
223 // results in a re-entrant call where the second call is one page behind.
224 for( int attempts = 0; attempts < 3; ++attempts )
225 m_notebook->ChangeSelection( ii );
226
227 return;
228 }
229 }
230
231 m_nestedTables.push_back( aTable );
232 AddTable( aTable.get(), tabTitle, true );
233
234 // Something is pretty fishy with wxAuiNotebook::ChangeSelection(); on Mac at least it
235 // results in a re-entrant call where the second call is one page behind.
236 for( int attempts = 0; attempts < 3; ++attempts )
237 m_notebook->ChangeSelection( m_notebook->GetPageCount() - 1 );
238}
239
240
241void PANEL_FP_LIB_TABLE::AddTable( LIBRARY_TABLE* aTable, const wxString& aTitle, bool aClosable )
242{
244 wxString projectPath = m_project->GetProjectPath();
245
247
248 WX_GRID* grid = get_grid( (int) m_notebook->GetPageCount() - 1 );
249
250 if( aTable->Path().StartsWith( projectPath ) )
251 {
252 grid->SetTable( new FP_LIB_TABLE_GRID_DATA_MODEL( m_parent, grid, *aTable, adapter, m_pluginChoices,
254 true /* take ownership */ );
255 }
256 else
257 {
258 wxString* lastGlobalLibDir = nullptr;
259
260 if( PCBNEW_SETTINGS* cfg = GetAppSettings<PCBNEW_SETTINGS>( "pcbnew" ) )
261 {
262 if( cfg->m_LastFootprintLibDir.IsEmpty() )
263 cfg->m_LastFootprintLibDir = PATHS::GetDefaultUserFootprintsPath();
264
265 lastGlobalLibDir = &cfg->m_LastFootprintLibDir;
266 }
267
268 grid->SetTable( new FP_LIB_TABLE_GRID_DATA_MODEL( m_parent, grid, *aTable, adapter, m_pluginChoices,
269 lastGlobalLibDir, wxEmptyString, m_supportedFpFiles ),
270 true /* take ownership */ );
271 }
272
273 static_cast<LIB_TABLE_GRID_DATA_MODEL*>( grid->GetTable() )->RecheckRows();
274
275 LIB_TABLE_NOTEBOOK_PANEL* notebookPanel =
276 static_cast<LIB_TABLE_NOTEBOOK_PANEL*>( m_notebook->GetPage( m_notebook->GetPageCount() - 1 ) );
277
278 static_cast<LIB_TABLE_GRID_DATA_MODEL*>( grid->GetTable() )
280 [notebookPanel]()
281 {
282 notebookPanel->MarkDirty();
283 } );
284
285 // add Cut, Copy, and Paste to wxGrids
286 grid->PushEventHandler( new FP_GRID_TRICKS( this, grid,
287 [this]( wxCommandEvent& event )
288 {
289 appendRowHandler( event );
290 } ) );
291
292 auto autoSizeCol =
293 [&]( int aCol )
294 {
295 int prevWidth = grid->GetColSize( aCol );
296
297 grid->AutoSizeColumn( aCol, false );
298 grid->SetColSize( aCol, std::max( prevWidth, grid->GetColSize( aCol ) ) );
299 };
300
301 // all but COL_OPTIONS, which is edited with Option Editor anyways.
302 autoSizeCol( COL_NICKNAME );
303 autoSizeCol( COL_TYPE );
304 autoSizeCol( COL_URI );
305 autoSizeCol( COL_DESCR );
306
307 if( grid->GetNumberRows() > 0 )
308 {
309 grid->SetGridCursor( 0, COL_NICKNAME );
310 grid->SelectRow( 0 );
311 }
312}
313
314
316 PANEL_FP_LIB_TABLE_BASE( aParent ),
317 m_project( aProject ),
318 m_parent( aParent ),
320{
321 m_lastProjectLibDir = m_project->GetProjectPath();
322
324
325 for( auto& [fileType, desc] : m_supportedFpFiles )
327
328 std::optional<LIBRARY_TABLE*> table = Pgm().GetLibraryManager().Table( LIBRARY_TABLE_TYPE::FOOTPRINT,
330 wxASSERT( table.has_value() );
331
332 AddTable( table.value(), _( "Global Libraries" ), false /* closable */ );
333
334 std::optional<LIBRARY_TABLE*> projectTable = Pgm().GetLibraryManager().Table( LIBRARY_TABLE_TYPE::FOOTPRINT,
336
337 if( projectTable.has_value() )
338 AddTable( projectTable.value(), _( "Project Specific Libraries" ), false /* closable */ );
339
340 m_notebook->SetArtProvider( new WX_AUI_TAB_ART() );
341
342 // add Cut, Copy, and Paste to wxGrids
343 m_path_subs_grid->PushEventHandler( new GRID_TRICKS( m_path_subs_grid ) );
344
346
347 m_path_subs_grid->SetColLabelValue( 0, _( "Name" ) );
348 m_path_subs_grid->SetColLabelValue( 1, _( "Value" ) );
349
350 // Configure button logos
356
357 // For aesthetic reasons, we must set the size of m_browseButton to match the other bitmaps
358 // manually (for instance m_append_button)
359 Layout(); // Needed at least on MSW to compute the actual buttons sizes, after initializing
360 // their bitmaps
361 wxSize buttonSize = m_append_button->GetSize();
362
363 m_browseButton->SetWidthPadding( 4 );
364 m_browseButton->SetMinSize( buttonSize );
365
366 // Populate the browse library options
367 wxMenu* browseMenu = m_browseButton->GetSplitButtonMenu();
368
369 for( auto& [type, desc] : m_supportedFpFiles )
370 {
371 wxString entryStr = PCB_IO_MGR::ShowType( type );
372 wxString midPart;
373
374 if( desc.m_IsFile && !desc.m_FileExtensions.empty() )
375 {
376 entryStr << wxString::Format( wxS( " (%s)" ), JoinExtensions( desc.m_FileExtensions ) );
377 }
378 else if( !desc.m_IsFile && !desc.m_ExtensionsInDir.empty() )
379 {
380 midPart = wxString::Format( _( "folder with %s files" ), JoinExtensions( desc.m_ExtensionsInDir ) );
381 entryStr << wxString::Format( wxS( " (%s)" ), midPart );
382 }
383
384 browseMenu->Append( type, entryStr );
385 browseMenu->Bind( wxEVT_COMMAND_MENU_SELECTED, &PANEL_FP_LIB_TABLE::browseLibrariesHandler, this, type );
386 }
387
388 Layout();
389
390 m_notebook->Bind( wxEVT_AUINOTEBOOK_PAGE_CLOSE, &PANEL_FP_LIB_TABLE::onNotebookPageCloseRequest, this );
391 m_notebook->Bind( wxEVT_AUINOTEBOOK_PAGE_CHANGING, &PANEL_FP_LIB_TABLE::onNotebookPageChangeRequest, this );
392 // This is the button only press for the browse button instead of the menu
394
395 m_parent->SetCanCloseCheck(
396 [this]()
397 {
398 for( int ii = 0; ii < (int) m_notebook->GetPageCount(); ++ii )
399 {
401 static_cast<LIB_TABLE_NOTEBOOK_PANEL*>( m_notebook->GetPage( ii ) );
402
403 if( panel->GetClosable() )
404 {
405 bool wasDirty = panel->TableModified();
406
407 if( !panel->GetCanClose() )
408 return false;
409
410 if( wasDirty && !panel->TableModified() )
411 {
412 m_parent->m_GlobalTableChanged = true;
413 m_parent->m_ProjectTableChanged = true;
414 }
415 }
416 }
417
418 return true;
419 } );
420}
421
422
424{
425 wxMenu* browseMenu = m_browseButton->GetSplitButtonMenu();
426
427 for( auto& [type, desc] : m_supportedFpFiles )
428 browseMenu->Unbind( wxEVT_COMMAND_MENU_SELECTED, &PANEL_FP_LIB_TABLE::browseLibrariesHandler, this, type );
429
430 m_browseButton->Unbind( wxEVT_BUTTON, &PANEL_FP_LIB_TABLE::browseLibrariesHandler, this );
431
432 // Delete the GRID_TRICKS.
433 // (Notebook page GRID_TRICKS are deleted by LIB_TABLE_NOTEBOOK_PANEL.)
434 m_path_subs_grid->PopEventHandler( true );
435}
436
437
439{
440 return static_cast<FP_LIB_TABLE_GRID_DATA_MODEL*>( get_grid( aPage )->GetTable() );
441}
442
443
445{
446 return static_cast<LIB_TABLE_NOTEBOOK_PANEL*>( m_notebook->GetPage( aPage ) )->GetGrid();
447}
448
449
451{
452 // No visibility control for footprint libraries yet; this feature is primarily
453 // useful for database libraries and it's only implemented for schematic symbols
454 // at the moment.
455 for( int page = 0 ; page < (int) m_notebook->GetPageCount(); ++page )
456 {
457 WX_GRID* grid = get_grid( page );
458 grid->HideCol( COL_VISIBLE );
459 }
460
461 // for ALT+A handling, we want the initial focus to be on the first selected grid.
462 m_parent->SetInitialFocus( cur_grid() );
463
464 return true;
465}
466
467
469{
470 for( const auto& plugin : PCB_IO_MGR::PLUGIN_REGISTRY::Instance()->AllPlugins() )
471 {
472 IO_RELEASER<PCB_IO> pi( plugin.m_createFunc() );
473
474 if( !pi )
475 continue;
476
477 if( const IO_BASE::IO_FILE_DESC& desc = pi->GetLibraryDesc() )
478 m_supportedFpFiles.emplace( plugin.m_type, desc );
479 }
480
482 IO_BASE::IO_FILE_DESC( _( "Table (nested library table)" ), {} ) );
483}
484
485
487{
488 for( int page = 0 ; page < (int) m_notebook->GetPageCount(); ++page )
489 {
490 WX_GRID* grid = get_grid( page );
491
493 [&]( int aRow, int aCol )
494 {
495 // show the tabbed panel holding the grid we have flunked:
496 if( m_notebook->GetSelection() != page )
497 m_notebook->SetSelection( page );
498
499 grid->MakeCellVisible( aRow, 0 );
500 grid->SetGridCursor( aRow, aCol );
501 } ) )
502 {
503 return false;
504 }
505 }
506
507 return true;
508}
509
510
515
516
521
522
527
528
533
534
535// @todo refactor this function into single location shared with PANEL_SYM_LIB_TABLE
536void PANEL_FP_LIB_TABLE::onMigrateLibraries( wxCommandEvent& event )
537{
538 if( !cur_grid()->CommitPendingChanges() )
539 return;
540
541 wxArrayInt selectedRows = cur_grid()->GetSelectedRows();
542
543 if( selectedRows.empty() && cur_grid()->GetGridCursorRow() >= 0 )
544 selectedRows.push_back( cur_grid()->GetGridCursorRow() );
545
546 wxArrayInt rowsToMigrate;
547 wxString kicadType = PCB_IO_MGR::ShowType( PCB_IO_MGR::KICAD_SEXP );
548 wxString msg;
549 DIALOG_HTML_REPORTER errorReporter( this );
550
551 for( int row : selectedRows )
552 {
553 if( cur_grid()->GetCellValue( row, COL_TYPE ) != kicadType )
554 rowsToMigrate.push_back( row );
555 }
556
557 if( rowsToMigrate.size() <= 0 )
558 {
559 wxMessageBox( _( "Select one or more rows containing libraries to save as current KiCad format." ) );
560 return;
561 }
562 else
563 {
564 if( rowsToMigrate.size() == 1 )
565 {
566 msg.Printf( _( "Save '%s' as current KiCad format and replace entry in table?" ),
567 cur_grid()->GetCellValue( rowsToMigrate[0], COL_NICKNAME ) );
568 }
569 else
570 {
571 msg.Printf( _( "Save %d libraries as current KiCad format and replace entries in table?" ),
572 (int) rowsToMigrate.size() );
573 }
574
575 if( !IsOK( m_parent, msg ) )
576 return;
577 }
578
579 for( int row : rowsToMigrate )
580 {
581 wxString relPath = cur_grid()->GetCellValue( row, COL_URI );
582 wxString resolvedPath = ExpandEnvVarSubstitutions( relPath, m_project );
583 wxFileName legacyLib( resolvedPath );
584
585 if( !legacyLib.Exists() )
586 {
587 msg.Printf( _( "Library '%s' not found." ), relPath );
588 DisplayErrorMessage( wxGetTopLevelParent( this ), msg );
589 continue;
590 }
591
592 wxFileName newLib( resolvedPath );
593 newLib.AppendDir( newLib.GetName() + "." + FILEEXT::KiCadFootprintLibPathExtension );
594 newLib.SetName( "" );
595 newLib.ClearExt();
596
597 if( newLib.DirExists() )
598 {
599 msg.Printf( _( "Folder '%s' already exists. Do you want overwrite any existing footprints?" ),
600 newLib.GetFullPath() );
601
602 switch( wxMessageBox( msg, _( "Migrate Library" ), wxYES_NO|wxCANCEL|wxICON_QUESTION, m_parent ) )
603 {
604 case wxYES: break;
605 case wxNO: continue;
606 case wxCANCEL: return;
607 }
608 }
609
610 wxString options = cur_grid()->GetCellValue( row, COL_OPTIONS );
611 std::map<std::string, UTF8> props( LIBRARY_TABLE::ParseOptions( options.ToStdString() ) );
612
613 if( PCB_IO_MGR::ConvertLibrary( props, legacyLib.GetFullPath(), newLib.GetFullPath(),
614 errorReporter.m_Reporter ) )
615 {
616 relPath = NormalizePath( newLib.GetFullPath(), &Pgm().GetLocalEnvVariables(), m_project );
617
618 cur_grid()->SetCellValue( row, COL_URI, relPath );
619 cur_grid()->SetCellValue( row, COL_TYPE, kicadType );
620 }
621 else
622 {
623 DisplayErrorMessage( m_parent, wxString::Format( _( "Failed to save footprint library file '%s'." ),
624 newLib.GetFullPath() ) );
625 }
626 }
627
628 if( errorReporter.m_Reporter->HasMessage() )
629 {
630 errorReporter.m_Reporter->Flush(); // Build HTML messages
631 errorReporter.ShowModal();
632 }
633}
634
635
637{
638 if( !cur_grid()->CommitPendingChanges() )
639 return;
640
642
643 // We are bound both to the menu and button with this one handler
644 // So we must set the file type based on it
645 if( event.GetEventType() == wxEVT_BUTTON )
646 {
647 // Let's default to adding a kicad footprint file for just the footprint
649 }
650 else
651 {
652 fileType = static_cast<PCB_IO_MGR::PCB_FILE_T>( event.GetId() );
653 }
654
656 return;
657
658 const ENV_VAR_MAP& envVars = Pgm().GetLocalEnvVariables();
659 const IO_BASE::IO_FILE_DESC& fileDesc = m_supportedFpFiles.at( fileType );
661
662 wxString title;
663 wxString dummy;
664 wxString* lastDir;
665
667 title = _( "Select Library Table" );
668 else
669 title = wxString::Format( _( "Select %s Library" ), PCB_IO_MGR::ShowType( fileType ) );
670
671 if( m_notebook->GetSelection() == 0 )
672 lastDir = cfg ? &cfg->m_LastFootprintLibDir : &dummy;
673 else
674 lastDir = &m_lastProjectLibDir;
675
676 wxArrayString files;
677
678 if( fileDesc.m_IsFile )
679 {
680 wxFileDialog dlg( m_parent, title, *lastDir, wxEmptyString, fileDesc.FileFilter(),
681 wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE );
682
684
685 if( dlg.ShowModal() == wxID_CANCEL )
686 return;
687
688 dlg.GetPaths( files );
689 *lastDir = dlg.GetDirectory();
690 }
691 else
692 {
693 wxDirDialog dlg( m_parent, title, *lastDir, wxDD_DEFAULT_STYLE | wxDD_DIR_MUST_EXIST | wxDD_MULTIPLE );
694
695 if( dlg.ShowModal() == wxID_CANCEL )
696 return;
697
698 dlg.GetPaths( files );
699
700 if( !files.IsEmpty() )
701 {
702 wxFileName first( files.front() );
703 *lastDir = first.GetPath();
704 }
705 }
706
707 // Drop the last directory if the path is a .pretty folder
709 cfg->m_LastFootprintLibDir = cfg->m_LastFootprintLibDir.BeforeLast( wxFileName::GetPathSeparator() );
710
711
712 bool addDuplicates = false;
713 bool applyToAll = false;
714 wxString warning = _( "Warning: Duplicate Nicknames" );
715 wxString msg = _( "An item nicknamed '%s' already exists." );
716 wxString detailedMsg = _( "One of the nicknames will need to be changed." );
717
718 for( const wxString& filePath : files )
719 {
720 wxFileName fn( filePath );
721 wxString nickname = LIB_ID::FixIllegalChars( fn.GetName(), true );
722 bool doAdd = true;
723
726 {
727 nickname = LIB_ID::FixIllegalChars( fn.GetFullName(), true ).wx_str();
728 }
729
730 if( cur_model()->ContainsNickname( nickname ) )
731 {
732 if( !applyToAll )
733 {
734 // The cancel button adds the library to the table anyway
735 addDuplicates = OKOrCancelDialog( m_parent, warning, wxString::Format( msg, nickname ), detailedMsg,
736 _( "Skip" ), _( "Add Anyway" ), &applyToAll ) == wxID_CANCEL;
737 }
738
739 doAdd = addDuplicates;
740 }
741
742 if( doAdd && cur_grid()->AppendRows( 1 ) )
743 {
744 int last_row = cur_grid()->GetNumberRows() - 1;
745
746 cur_grid()->SetCellValue( last_row, COL_NICKNAME, nickname );
747 cur_grid()->SetCellValue( last_row, COL_TYPE, PCB_IO_MGR::ShowType( fileType ) );
748
749 // try to use path normalized to an environmental variable or project path
750 wxString path = NormalizePath( filePath, &envVars, m_project->GetProjectPath() );
751
752 // Do not use the project path in the global library table. This will almost
753 // assuredly be wrong for a different project.
754 if( m_notebook->GetSelection() == 0 && path.Contains( wxT( "${KIPRJMOD}" ) ) )
755 path = fn.GetFullPath();
756
757 cur_grid()->SetCellValue( last_row, COL_URI, path );
758 }
759 }
760
761 if( !files.IsEmpty() )
762 {
763 cur_grid()->MakeCellVisible( cur_grid()->GetNumberRows() - 1, COL_ENABLED );
764 cur_grid()->SetGridCursor( cur_grid()->GetNumberRows() - 1, COL_NICKNAME );
765 }
766}
767
768
769void PANEL_FP_LIB_TABLE::onNotebookPageCloseRequest( wxAuiNotebookEvent& aEvent )
770{
771 wxAuiNotebook* notebook = (wxAuiNotebook*) aEvent.GetEventObject();
772 wxWindow* page = notebook->GetPage( aEvent.GetSelection() );
773
774 if( LIB_TABLE_NOTEBOOK_PANEL* panel = dynamic_cast<LIB_TABLE_NOTEBOOK_PANEL*>( page ) )
775 {
776 if( panel->GetClosable() )
777 {
778 if( !panel->GetCanClose() )
779 aEvent.Veto();
780 }
781 else
782 {
783 aEvent.Veto();
784 }
785 }
786}
787
788
790{
791 // Account for scroll bars
792 aWidth -= ( m_path_subs_grid->GetSize().x - m_path_subs_grid->GetClientSize().x );
793
794 m_path_subs_grid->AutoSizeColumn( 0 );
795 m_path_subs_grid->SetColSize( 0, std::max( 72, m_path_subs_grid->GetColSize( 0 ) ) );
796 m_path_subs_grid->SetColSize( 1, std::max( 120, aWidth - m_path_subs_grid->GetColSize( 0 ) ) );
797}
798
799
800void PANEL_FP_LIB_TABLE::onSizeGrid( wxSizeEvent& event )
801{
802 adjustPathSubsGridColumns( event.GetSize().GetX() );
803
804 event.Skip();
805}
806
807
808void PANEL_FP_LIB_TABLE::onReset( wxCommandEvent& event )
809{
810 if( !cur_grid()->CommitPendingChanges() )
811 return;
812
813 WX_GRID* grid = get_grid( 0 );
814
815 // No need to prompt to preserve an empty table
816 if( grid->GetNumberRows() > 0 && !IsOK( this, wxString::Format( _( "This action will reset your global library "
817 "table on disk and cannot be undone." ) ) ) )
818 {
819 return;
820 }
821
822 wxString* lastGlobalLibDir = nullptr;
823
824 if( PCBNEW_SETTINGS* cfg = GetAppSettings<PCBNEW_SETTINGS>( "pcbnew" ) )
825 {
826 if( cfg->m_LastFootprintLibDir.IsEmpty() )
827 cfg->m_LastFootprintLibDir = PATHS::GetDefaultUserFootprintsPath();
828
829 lastGlobalLibDir = &cfg->m_LastFootprintLibDir;
830 }
831
833
834 // Go ahead and reload here because this action takes place even if the dialog is canceled
836
837 if( KIFACE *face = m_parent->Kiway().KiFACE( KIWAY::FACE_PCB ) )
838 face->PreloadLibraries( &m_parent->Kiway() );
839
840 grid->Freeze();
841
842 wxGridTableBase* table = grid->GetTable();
843 grid->DestroyTable( table );
844
845 std::optional<LIBRARY_TABLE*> newTable = Pgm().GetLibraryManager().Table( LIBRARY_TABLE_TYPE::FOOTPRINT,
847 wxASSERT( newTable );
848
850
851 grid->SetTable( new FP_LIB_TABLE_GRID_DATA_MODEL( m_parent, grid, *newTable.value(), adapter, m_pluginChoices,
852 lastGlobalLibDir, wxEmptyString, m_supportedFpFiles ),
853 true /* take ownership */ );
854
855 m_parent->m_GlobalTableChanged = true;
856
857 grid->Thaw();
858
859 if( grid->GetNumberRows() > 0 )
860 {
861 grid->SetGridCursor( 0, COL_NICKNAME );
862 grid->SelectRow( 0 );
863 }
864}
865
866
867void PANEL_FP_LIB_TABLE::onNotebookPageChangeRequest( wxAuiNotebookEvent& aEvent )
868{
870 aEvent.Veto();
871 else
872 aEvent.Skip();
873}
874
875
876void PANEL_FP_LIB_TABLE::onPageChange( wxAuiNotebookEvent& event )
877{
878 m_resetGlobal->Enable( m_notebook->GetSelection() == 0 );
879}
880
881
883{
884 if( !cur_grid()->CommitPendingChanges() )
885 return false;
886
887 if( !verifyTables() )
888 return false;
889
890 std::optional<LIBRARY_TABLE*> optTable = Pgm().GetLibraryManager().Table( LIBRARY_TABLE_TYPE::FOOTPRINT,
892 wxCHECK( optTable, false );
893 LIBRARY_TABLE* globalTable = *optTable;
894
895 if( get_model( 0 )->Table() != *globalTable )
896 {
897 m_parent->m_GlobalTableChanged = true;
898 *globalTable = get_model( 0 )->Table();
899
900 globalTable->Save().map_error(
901 []( const LIBRARY_ERROR& aError )
902 {
903 wxMessageBox( _( "Error saving global library table:\n\n" ) + aError.message,
904 _( "File Save Error" ), wxOK | wxICON_ERROR );
905 } );
906 }
907
909
910 if( optTable.has_value() && get_model( 1 )->Table().Path() == optTable.value()->Path() )
911 {
912 LIBRARY_TABLE* projectTable = *optTable;
913
914 if( get_model( 1 )->Table() != *projectTable )
915 {
916 m_parent->m_ProjectTableChanged = true;
917 *projectTable = get_model( 1 )->Table();
918
919 projectTable->Save().map_error(
920 []( const LIBRARY_ERROR& aError )
921 {
922 wxMessageBox( _( "Error saving project library table:\n\n" ) + aError.message,
923 _( "File Save Error" ), wxOK | wxICON_ERROR );
924 } );
925 }
926 }
927
929 return true;
930}
931
932
936{
937 wxRegEx re( ".*?(\\$\\{(.+?)\\})|(\\$\\((.+?)\\)).*?", wxRE_ADVANCED );
938 wxASSERT( re.IsValid() ); // wxRE_ADVANCED is required.
939
940 std::set< wxString > unique;
941
942 // clear the table
943 m_path_subs_grid->ClearRows();
944
945 for( int page = 0 ; page < (int) m_notebook->GetPageCount(); ++page )
946 {
948
949 for( int row = 0; row < model->GetNumberRows(); ++row )
950 {
951 wxString uri = model->GetValue( row, COL_URI );
952
953 while( re.Matches( uri ) )
954 {
955 wxString envvar = re.GetMatch( uri, 2 );
956
957 // if not ${...} form then must be $(...)
958 if( envvar.IsEmpty() )
959 envvar = re.GetMatch( uri, 4 );
960
961 // ignore duplicates
962 unique.insert( envvar );
963
964 // delete the last match and search again
965 uri.Replace( re.GetMatch( uri, 0 ), wxEmptyString );
966 }
967 }
968 }
969
970 // Make sure this special environment variable shows up even if it was not used yet. It is
971 // automatically set by KiCad to the directory holding the current project.
972 unique.insert( PROJECT_VAR_NAME );
973 unique.insert( ENV_VAR::GetVersionedEnvVarName( wxS( "FOOTPRINT_DIR" ) ) );
974
975 // This special environment variable is used to locate 3d shapes
976 unique.insert( ENV_VAR::GetVersionedEnvVarName( wxS( "3DMODEL_DIR" ) ) );
977
978 for( const wxString& evName : unique )
979 {
980 int row = m_path_subs_grid->GetNumberRows();
981 m_path_subs_grid->AppendRows( 1 );
982
983 m_path_subs_grid->SetCellValue( row, 0, wxT( "${" ) + evName + wxT( "}" ) );
984 m_path_subs_grid->SetCellEditor( row, 0, new GRID_CELL_READONLY_TEXT_EDITOR() );
985
986 wxString evValue;
987 wxGetEnv( evName, &evValue );
988 m_path_subs_grid->SetCellValue( row, 1, evValue );
989 m_path_subs_grid->SetCellEditor( row, 1, new GRID_CELL_READONLY_TEXT_EDITOR() );
990 }
991
992 adjustPathSubsGridColumns( m_path_subs_grid->GetRect().GetWidth() );
993}
994
995
996//-----</event handlers>---------------------------------
997
998
999
1000void InvokePcbLibTableEditor( KIWAY* aKiway, wxWindow* aCaller )
1001{
1002 DIALOG_EDIT_LIBRARY_TABLES dlg( aCaller, _( "Footprint Libraries" ) );
1003 dlg.SetKiway( &dlg, aKiway );
1004
1005 dlg.InstallPanel( new PANEL_FP_LIB_TABLE( &dlg, &aKiway->Prj() ) );
1006
1007 if( dlg.ShowModal() == wxID_CANCEL )
1008 return;
1009
1010 if( dlg.m_GlobalTableChanged )
1012
1013 if( dlg.m_ProjectTableChanged )
1014 {
1015 // Trigger a reload of the table and cancel an in-progress background load
1017 }
1018
1019 // Trigger a reload in case any libraries have been added or removed
1020 if( KIFACE *face = aKiway->KiFACE( KIWAY::FACE_PCB ) )
1021 face->PreloadLibraries( aKiway );
1022
1023 std::string payload = "";
1026 aKiway->ExpressMail( FRAME_CVPCB, MAIL_RELOAD_LIB, payload );
1027}
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition bitmap.cpp:110
Class DIALOG_HTML_REPORTER.
WX_HTML_REPORT_BOX * m_Reporter
An options editor in the form of a two column name/value spreadsheet like (table) UI.
Dialog helper object to sit in the inheritance tree between wxDialog and any class written by wxFormB...
Definition dialog_shim.h:69
int ShowModal() override
An interface to the global shared library manager that is schematic-specific and linked to one projec...
void openTable(const LIBRARY_TABLE_ROW &aRow) override
FP_GRID_TRICKS(PANEL_FP_LIB_TABLE *aPanel, WX_GRID *aGrid, std::function< void(wxCommandEvent &)> aAddHandler)
static bool SupportsVisibilityColumn()
wxString getTablePreamble() override
void optionsEditor(int aRow) override
PANEL_FP_LIB_TABLE * m_panel
bool supportsVisibilityColumn() override
This class builds a wxGridTableBase by wrapping an #FP_LIB_TABLE object.
FP_LIB_TABLE_GRID_DATA_MODEL(DIALOG_SHIM *aParent, WX_GRID *aGrid, const LIBRARY_TABLE &aTableToEdit, FOOTPRINT_LIBRARY_ADAPTER *aAdapter, const wxArrayString &aPluginChoices, wxString *aMRUDirectory, const wxString &aProjectPath, const std::map< PCB_IO_MGR::PCB_FILE_T, IO_BASE::IO_FILE_DESC > &aSupportedFiles)
wxString getFileTypes(WX_GRID *aGrid, int aRow) override
void SetValue(int aRow, int aCol, const wxString &aValue) override
const std::map< PCB_IO_MGR::PCB_FILE_T, IO_BASE::IO_FILE_DESC > & m_supportedFpFiles
Add mouse and command handling (such as cut, copy, and paste) to a WX_GRID instance.
Definition grid_tricks.h:61
WX_GRID * m_grid
I don't own the grid, but he owns me.
void SetTooltipEnable(int aCol, bool aEnable=true)
Enable the tooltip for a column.
Definition grid_tricks.h:75
void SetKiway(wxWindow *aDest, KIWAY *aKiway)
It is only used for debugging, since "this" is not a wxWindow*.
A minimalistic software bus for communications between various DLLs/DSOs (DSOs) within the same KiCad...
Definition kiway.h:315
virtual void ExpressMail(FRAME_T aDestination, MAIL_T aCommand, std::string &aPayload, wxWindow *aSource=nullptr, bool aFromOtherThread=false)
Send aPayload to aDestination from aSource.
Definition kiway.cpp:500
virtual KIFACE * KiFACE(FACE_T aFaceId, bool doLoad=true)
Return the KIFACE* given a FACE_T.
Definition kiway.cpp:211
@ FACE_PCB
pcbnew DSO
Definition kiway.h:323
virtual PROJECT & Prj() const
Return the PROJECT associated with this KIWAY.
Definition kiway.cpp:205
static wxString ExpandURI(const wxString &aShortURI, const PROJECT &aProject)
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...
void ApplyLibOverrides(LIBRARY_TABLE &aTable)
Applies stored user overrides (disabled/hidden) to rows of a read-only table.
static bool CreateGlobalTable(LIBRARY_TABLE_TYPE aType, bool aPopulateDefaultLibraries)
void LoadGlobalTables(std::initializer_list< LIBRARY_TABLE_TYPE > aTablesToLoad={})
(Re)loads the global library tables in the given list, or all tables if no list is given
void ProjectChanged()
Notify all adapters that the project has changed.
void SetOptions(const wxString &aOptions)
const wxString & Type() const
static const wxString TABLE_TYPE_NAME
const wxString & URI() const
const wxString & Nickname() const
const wxString & Options() const
LIBRARY_RESULT< void > Save()
static std::map< std::string, UTF8 > ParseOptions(const std::string &aOptionsList)
const wxString & Path() const
static UTF8 FixIllegalChars(const UTF8 &aLibItemName, bool aLib)
Replace illegal LIB_ID item name characters with underscores '_'.
Definition lib_id.cpp:192
This abstract base class mixes any object derived from #LIB_TABLE into wxGridTableBase so the result ...
virtual LIBRARY_TABLE_ROW & at(size_t aIndex)
void SetChangeCallback(std::function< void()> aCallback)
LIBRARY_TABLE_ROW & At(size_t aIndex)
void SetValue(int aRow, int aCol, const wxString &aValue) override
LIB_TABLE_GRID_DATA_MODEL(DIALOG_SHIM *aParent, WX_GRID *aGrid, const LIBRARY_TABLE &aTableToEdit, LIBRARY_MANAGER_ADAPTER *aAdapter, const wxArrayString &aPluginChoices, wxString *aMRUDirectory, const wxString &aProjectPath)
static void MoveUpHandler(WX_GRID *aGrid)
LIB_TABLE_GRID_TRICKS(WX_GRID *aGrid)
static void DeleteRowHandler(WX_GRID *aGrid)
static void AppendRowHandler(WX_GRID *aGrid)
static bool VerifyTable(WX_GRID *aGrid, bool aSupportsVisibilityColumn, std::function< void(int aRow, int aCol)> aErrorHandler)
static void MoveDownHandler(WX_GRID *aGrid)
static void AddTable(wxAuiNotebook *aNotebook, const wxString &aTitle, bool aClosable)
STD_BITMAP_BUTTON * m_move_up_button
STD_BITMAP_BUTTON * m_append_button
STD_BITMAP_BUTTON * m_move_down_button
STD_BITMAP_BUTTON * m_delete_button
PANEL_FP_LIB_TABLE_BASE(wxWindow *parent, wxWindowID id=wxID_ANY, const wxPoint &pos=wxDefaultPosition, const wxSize &size=wxSize(-1,-1), long style=wxTAB_TRAVERSAL, const wxString &name=wxEmptyString)
Dialog to show and edit symbol library tables.
void OpenTable(const std::shared_ptr< LIBRARY_TABLE > &table, const wxString &aTitle)
bool verifyTables()
Trim important fields, removes blank row entries, and checks for duplicates.
std::map< PCB_IO_MGR::PCB_FILE_T, IO_BASE::IO_FILE_DESC > m_supportedFpFiles
void onSizeGrid(wxSizeEvent &event) override
void moveUpHandler(wxCommandEvent &event) override
bool TransferDataToWindow() override
WX_GRID * cur_grid() const
void adjustPathSubsGridColumns(int aWidth)
void moveDownHandler(wxCommandEvent &event) override
wxArrayString m_pluginChoices
PANEL_FP_LIB_TABLE(DIALOG_EDIT_LIBRARY_TABLES *aParent, PROJECT *aProject)
void populateEnvironReadOnlyTable()
Populate the readonly environment variable table with names and values by examining all the full_uri ...
void deleteRowHandler(wxCommandEvent &event) override
void onNotebookPageCloseRequest(wxAuiNotebookEvent &aEvent)
FP_LIB_TABLE_GRID_DATA_MODEL * get_model(int aPage) const
FP_LIB_TABLE_GRID_DATA_MODEL * cur_model() const
void onReset(wxCommandEvent &event) override
std::vector< std::shared_ptr< LIBRARY_TABLE > > m_nestedTables
void onMigrateLibraries(wxCommandEvent &event) override
void browseLibrariesHandler(wxCommandEvent &event)
void onPageChange(wxAuiNotebookEvent &event) override
bool TransferDataFromWindow() override
void AddTable(LIBRARY_TABLE *table, const wxString &aTitle, bool aClosable)
WX_GRID * get_grid(int aPage) const
void onNotebookPageChangeRequest(wxAuiNotebookEvent &aEvent)
DIALOG_EDIT_LIBRARY_TABLES * m_parent
void appendRowHandler(wxCommandEvent &event) override
static wxString GetDefaultUserFootprintsPath()
Gets the default path we point users to create projects.
Definition paths.cpp:93
wxString m_LastFootprintLibDir
static PLUGIN_REGISTRY * Instance()
Definition pcb_io_mgr.h:100
static bool ConvertLibrary(const std::map< std::string, UTF8 > &aOldFileProps, const wxString &aOldFilePath, const wxString &aNewFilePath, REPORTER *aReporter)
Convert a schematic symbol library to the latest KiCad format.
static PCB_FILE_T EnumFromStr(const wxString &aFileType)
Return the PCB_FILE_T from the corresponding plugin type name: "kicad", "legacy", etc.
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
@ PCB_FILE_UNKNOWN
0 is not a legal menu id on Mac
Definition pcb_io_mgr.h:57
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 ENV_VAR_MAP & GetLocalEnvVariables() const
Definition pgm_base.cpp:787
virtual LIBRARY_MANAGER & GetLibraryManager() const
Definition pgm_base.h:132
static FOOTPRINT_LIBRARY_ADAPTER * FootprintLibAdapter(PROJECT *aProject)
Container for project specific data.
Definition project.h:66
wxString wx_str() const
Definition utf8.cpp:45
void Flush()
Build the HTML messages page.
bool HasMessage() const override
Returns true if any messages were reported.
const wxString ExpandEnvVarSubstitutions(const wxString &aString, const PROJECT *aProject)
Replace any environment variable & text variable references with their values.
Definition common.cpp:708
wxString JoinExtensions(const std::vector< std::string > &aExts)
Join a list of file extensions for use in a file dialog.
Definition common.cpp:798
The common library.
int OKOrCancelDialog(wxWindow *aParent, const wxString &aWarning, const wxString &aMessage, const wxString &aDetailedMessage, const wxString &aOKLabel, const wxString &aCancelLabel, bool *aApplyToAll)
Display a warning dialog with aMessage and returns the user response.
Definition confirm.cpp:169
bool IsOK(wxWindow *aParent, const wxString &aMessage)
Display a yes/no dialog with aMessage and returns the user response.
Definition confirm.cpp:278
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition confirm.cpp:221
This file is part of the common library.
#define _(s)
Declaration of the eda_3d_viewer class.
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.
Functions related to environment variables, including help functions.
@ FRAME_FOOTPRINT_VIEWER
Definition frame_type.h:45
@ FRAME_FOOTPRINT_EDITOR
Definition frame_type.h:43
@ FRAME_CVPCB
Definition frame_type.h:52
static const std::string FootprintLibraryTableFileName
static const std::string KiCadFootprintLibPathExtension
std::map< wxString, ENV_VAR_ITEM > ENV_VAR_MAP
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:669
This file contains miscellaneous commonly used macros and functions.
@ MAIL_RELOAD_LIB
Definition mail_type.h:58
KICOMMON_API wxString GetVersionedEnvVarName(const wxString &aBaseName)
Construct a versioned environment variable based on this KiCad major version.
Definition env_vars.cpp:77
void AllowNetworkFileSystems(wxDialog *aDialog)
Configure a file dialog to show network and virtual file systems.
Definition wxgtk/ui.cpp:448
void InvokePcbLibTableEditor(KIWAY *aKiway, wxWindow *aCaller)
Function InvokePcbLibTableEditor shows the modal DIALOG_FP_LIB_TABLE for purposes of editing the glob...
PGM_BASE & Pgm()
The global program "get" accessor.
see class PGM_BASE
#define PROJECT_VAR_NAME
A variable name whose value holds the current project directory.
Definition project.h:41
T * GetAppSettings(const char *aFilename)
std::vector< FAB_LAYER_COLOR > dummy
MODEL3D_FORMAT_TYPE fileType(const char *aFileName)
Container that describes file type info.
Definition io_base.h:43
bool m_IsFile
Whether the library is a folder or a file.
Definition io_base.h:51
wxString FileFilter() const
Definition io_base.cpp:40
Implement a participant in the KIWAY alchemy.
Definition kiway.h:156
wxString message
std::string path
KIBIS_MODEL * model
std::vector< std::vector< std::string > > table
wxString result
Test unit parsing edge cases and error handling.
Definition of file extensions used in Kicad.