KiCad PCB EDA Suite
Loading...
Searching...
No Matches
panel_design_block_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 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
25#include <set>
26#include <wx/dir.h>
27#include <wx/regex.h>
28#include <wx/dirdlg.h>
29#include <wx/filedlg.h>
30#include <wx/msgdlg.h>
31
32#include <common.h>
33#include <project.h>
34#include <env_vars.h>
35#include <lib_id.h>
36#include <bitmaps.h>
38#include <widgets/wx_grid.h>
41#include <confirm.h>
44#include <pgm_base.h>
45#include <env_paths.h>
50#include <kiway.h>
51#include <kiplatform/ui.h>
52#include <kiway_mail.h>
55#include <paths.h>
56#include <macros.h>
60
73
74
79class LIBRARY_TRAVERSER : public wxDirTraverser
80{
81public:
82 LIBRARY_TRAVERSER( std::vector<std::string> aSearchExtensions, wxString aInitialDir ) :
83 m_searchExtensions( aSearchExtensions ), m_currentDir( aInitialDir )
84 {
85 }
86
87 virtual wxDirTraverseResult OnFile( const wxString& aFileName ) override
88 {
89 wxFileName file( aFileName );
90
91 for( const std::string& ext : m_searchExtensions )
92 {
93 if( file.GetExt().IsSameAs( ext, false ) )
94 m_foundDirs.insert( { m_currentDir, 1 } );
95 }
96
97 return wxDIR_CONTINUE;
98 }
99
100 virtual wxDirTraverseResult OnOpenError( const wxString& aOpenErrorName ) override
101 {
102 m_failedDirs.insert( { aOpenErrorName, 1 } );
103 return wxDIR_IGNORE;
104 }
105
106 bool HasDirectoryOpenFailures() { return m_failedDirs.size() > 0; }
107
108 virtual wxDirTraverseResult OnDir( const wxString& aDirName ) override
109 {
110 m_currentDir = aDirName;
111 return wxDIR_CONTINUE;
112 }
113
114 void GetPaths( wxArrayString& aPathArray )
115 {
116 for( std::pair<const wxString, int>& foundDirsPair : m_foundDirs )
117 aPathArray.Add( foundDirsPair.first );
118 }
119
120 void GetFailedPaths( wxArrayString& aPathArray )
121 {
122 for( std::pair<const wxString, int>& failedDirsPair : m_failedDirs )
123 aPathArray.Add( failedDirsPair.first );
124 }
125
126private:
127 std::vector<std::string> m_searchExtensions;
128 wxString m_currentDir;
129 std::unordered_map<wxString, int> m_foundDirs;
130 std::unordered_map<wxString, int> m_failedDirs;
131};
132
133
138{
139public:
141 LIBRARY_MANAGER_ADAPTER* aAdapter, const wxArrayString& aPluginChoices,
142 wxString* aMRUDirectory, const wxString& aProjectPath,
144 IO_BASE::IO_FILE_DESC>& aSupportedFiles ) :
145 LIB_TABLE_GRID_DATA_MODEL( aParent, aGrid, aTableToEdit, aAdapter, aPluginChoices, aMRUDirectory,
146 aProjectPath ),
147 m_supportedDesignBlockFiles( aSupportedFiles )
148 {
149 }
150
151 void SetValue( int aRow, int aCol, const wxString& aValue ) override
152 {
153 wxCHECK( aRow < (int) size(), /* void */ );
154
155 LIB_TABLE_GRID_DATA_MODEL::SetValue( aRow, aCol, aValue );
156
157 // If setting a filepath, attempt to auto-detect the format
158 if( aCol == COL_URI )
159 {
160 LIBRARY_TABLE_ROW& row = at( static_cast<size_t>( aRow ) );
161 wxString uri = LIBRARY_MANAGER::ExpandURI( row.URI(), Pgm().GetSettingsManager().Prj() );
162
165
166 if( pluginType != DESIGN_BLOCK_IO_MGR::FILE_TYPE_NONE )
167 SetValue( aRow, COL_TYPE, DESIGN_BLOCK_IO_MGR::ShowType( pluginType ) );
168 }
169 }
170
171protected:
172 wxString getFileTypes( WX_GRID* aGrid, int aRow ) override
173 {
174 auto* table = static_cast<DESIGN_BLOCK_LIB_TABLE_GRID_DATA_MODEL*>( aGrid->GetTable() );
175 LIBRARY_TABLE_ROW& tableRow = table->at( aRow );
176
177 if( tableRow.Type() == LIBRARY_TABLE_ROW::TABLE_TYPE_NAME )
178 {
179 wxString filter = _( "Design Block Library Tables" );
180#ifndef __WXOSX__
181 filter << wxString::Format( _( " (%s)|%s" ), FILEEXT::DesignBlockLibraryTableFileName,
183#else
184 filter << wxString::Format( _( " (%s)|%s" ), wxFileSelectorDefaultWildcardStr,
185 wxFileSelectorDefaultWildcardStr );
186#endif
187 return filter;
188 }
189
190 if( tableRow.Type().IsEmpty() )
191 return wxEmptyString;
192
195 return wxEmptyString;
196
198
199 if( pluginDesc.m_IsFile )
200 return pluginDesc.FileFilter();
201
202 return wxEmptyString;
203 }
204
205private:
206 const std::map<DESIGN_BLOCK_IO_MGR::DESIGN_BLOCK_FILE_T, IO_BASE::IO_FILE_DESC>& m_supportedDesignBlockFiles;
207};
208
209
211{
212public:
219
221 {
222 return false;
223 }
224
225protected:
226 void optionsEditor( int aRow ) override
227 {
228 LIB_TABLE_GRID_DATA_MODEL* tbl = static_cast<LIB_TABLE_GRID_DATA_MODEL*>( m_grid->GetTable() );
229
230 if( tbl->GetNumberRows() > aRow )
231 {
232 LIBRARY_TABLE_ROW& row = tbl->At( static_cast<size_t>( aRow ) );
233 const wxString& options = row.Options();
234 wxString result = options;
235 std::map<std::string, UTF8> choices;
236
239 pi->GetLibraryOptions( &choices );
240
241 DIALOG_PLUGIN_OPTIONS dlg( wxGetTopLevelParent( m_grid ), row.Nickname(), choices, options, &result );
242 dlg.ShowModal();
243
244 if( options != result )
245 {
246 row.SetOptions( result );
247 m_grid->Refresh();
248 }
249 }
250 }
251
252 void openTable( const LIBRARY_TABLE_ROW& aRow ) override
253 {
254 wxFileName fn( LIBRARY_MANAGER::ExpandURI( aRow.URI(), Pgm().GetSettingsManager().Prj() ) );
255 std::shared_ptr<LIBRARY_TABLE> child = std::make_shared<LIBRARY_TABLE>( fn, LIBRARY_TABLE_SCOPE::GLOBAL, LIBRARY_TABLE_TYPE::DESIGN_BLOCK );
256
258
259 m_panel->OpenTable( child, aRow.Nickname() );
260 }
261
262 wxString getTablePreamble() override
263 {
264 return wxT( "(design_block_lib_table" );
265 }
266
271
272protected:
274};
275
276
277void PANEL_DESIGN_BLOCK_LIB_TABLE::OpenTable( const std::shared_ptr<LIBRARY_TABLE>& aTable, const wxString& aTitle )
278{
279 wxString tabTitle = aTitle;
280
281 if( aTable->IsReadOnly() )
282 tabTitle += wxS( " " ) + _( "(read-only)" );
283
284 for( int ii = 2; ii < (int) m_notebook->GetPageCount(); ++ii )
285 {
286 if( m_notebook->GetPageText( ii ) == tabTitle )
287 {
288 // Something is pretty fishy with wxAuiNotebook::ChangeSelection(); on Mac at least it
289 // results in a re-entrant call where the second call is one page behind.
290 for( int attempts = 0; attempts < 3; ++attempts )
291 m_notebook->ChangeSelection( ii );
292
293 return;
294 }
295 }
296
297 m_nestedTables.push_back( aTable );
298 AddTable( aTable.get(), tabTitle, true );
299
300 // Something is pretty fishy with wxAuiNotebook::ChangeSelection(); on Mac at least it
301 // results in a re-entrant call where the second call is one page behind.
302 for( int attempts = 0; attempts < 3; ++attempts )
303 m_notebook->ChangeSelection( m_notebook->GetPageCount() - 1 );
304}
305
306
307void PANEL_DESIGN_BLOCK_LIB_TABLE::AddTable( LIBRARY_TABLE* table, const wxString& aTitle, bool aClosable )
308{
309 DESIGN_BLOCK_LIBRARY_ADAPTER* adapter = m_project->DesignBlockLibs();
310 wxString projectPath = m_project->GetProjectPath();
311
313
314 WX_GRID* grid = get_grid( (int) m_notebook->GetPageCount() - 1 );
315
316 if( table->Path().StartsWith( projectPath ) )
317 {
319 &m_lastProjectLibDir, projectPath,
321 true /* take ownership */ );
322 }
323 else
324 {
325 wxString* lastGlobalLibDir = nullptr;
326
327 if( KICAD_SETTINGS* cfg = GetAppSettings<KICAD_SETTINGS>( "kicad" ) )
328 {
329 if( cfg->m_lastDesignBlockLibDir.IsEmpty() )
330 cfg->m_lastDesignBlockLibDir = PATHS::GetDefaultUserDesignBlocksPath();
331
332 lastGlobalLibDir = &cfg->m_lastDesignBlockLibDir;
333 }
334
336 lastGlobalLibDir, wxEmptyString,
338 true /* take ownership */ );
339 }
340
341 static_cast<LIB_TABLE_GRID_DATA_MODEL*>( grid->GetTable() )->RecheckRows();
342
343 LIB_TABLE_NOTEBOOK_PANEL* notebookPanel =
344 static_cast<LIB_TABLE_NOTEBOOK_PANEL*>( m_notebook->GetPage( m_notebook->GetPageCount() - 1 ) );
345
346 static_cast<LIB_TABLE_GRID_DATA_MODEL*>( grid->GetTable() )
348 [notebookPanel]()
349 {
350 notebookPanel->MarkDirty();
351 } );
352
353 // add Cut, Copy, and Paste to wxGrids
354 grid->PushEventHandler( new DESIGN_BLOCK_GRID_TRICKS( this, grid ) );
355
356 auto autoSizeCol =
357 [&]( int aCol )
358 {
359 int prevWidth = grid->GetColSize( aCol );
360
361 grid->AutoSizeColumn( aCol, false );
362 grid->SetColSize( aCol, std::max( prevWidth, grid->GetColSize( aCol ) ) );
363 };
364
365 // all but COL_OPTIONS, which is edited with Option Editor anyways.
366 autoSizeCol( COL_NICKNAME );
367 autoSizeCol( COL_TYPE );
368 autoSizeCol( COL_URI );
369 autoSizeCol( COL_DESCR );
370
371 if( grid->GetNumberRows() > 0 )
372 {
373 grid->SetGridCursor( 0, COL_NICKNAME );
374 grid->SelectRow( 0 );
375 }
376}
377
378
380 PROJECT* aProject ) :
382 m_project( aProject ),
383 m_parent( aParent ),
385{
386 m_lastProjectLibDir = m_project->GetProjectPath();
387
389
390 for( auto& [fileType, desc] : m_supportedDesignBlockFiles )
392
393 std::optional<LIBRARY_TABLE*> table = Pgm().GetLibraryManager().Table( LIBRARY_TABLE_TYPE::DESIGN_BLOCK,
395 wxASSERT( table.has_value() );
396
397 AddTable( table.value(), _( "Global Libraries" ), false /* closable */ );
398
399 std::optional<LIBRARY_TABLE*> projectTable = Pgm().GetLibraryManager().Table( LIBRARY_TABLE_TYPE::DESIGN_BLOCK,
401
402 if( projectTable.has_value() )
403 AddTable( projectTable.value(), _( "Project Specific Libraries" ), false /* closable */ );
404
405 m_notebook->SetArtProvider( new WX_AUI_TAB_ART() );
406
407 // There aren't (yet) any legacy DesignBlock libraries to migrate
408 m_migrate_libs_button->Hide();
409
410 // add Cut, Copy, and Paste to wxGrids
411 m_path_subs_grid->PushEventHandler( new GRID_TRICKS( m_path_subs_grid ) );
412
414
415 m_path_subs_grid->SetColLabelValue( 0, _( "Name" ) );
416 m_path_subs_grid->SetColLabelValue( 1, _( "Value" ) );
417
418 // Configure button logos
424
425 // For aesthetic reasons, we must set the size of m_browseButton to match the other bitmaps
426 // manually (for instance m_append_button)
427 Layout(); // Needed at least on MSW to compute the actual buttons sizes, after initializing
428 // their bitmaps
429 wxSize buttonSize = m_append_button->GetSize();
430
431 m_browseButton->SetWidthPadding( 4 );
432 m_browseButton->SetMinSize( buttonSize );
433
434 // Populate the browse library options
435 wxMenu* browseMenu = m_browseButton->GetSplitButtonMenu();
436
437 for( auto& [type, desc] : m_supportedDesignBlockFiles )
438 {
439 wxString entryStr = DESIGN_BLOCK_IO_MGR::ShowType( type );
440 wxString midPart;
441
442 if( desc.m_IsFile && !desc.m_FileExtensions.empty() )
443 {
444 entryStr << wxString::Format( wxS( " (%s)" ), JoinExtensions( desc.m_FileExtensions ) );
445 }
446 else if( !desc.m_IsFile && !desc.m_ExtensionsInDir.empty() )
447 {
448 midPart = wxString::Format( _( "folder with %s files" ), JoinExtensions( desc.m_ExtensionsInDir ) );
449 entryStr << wxString::Format( wxS( " (%s)" ), midPart );
450 }
451
452 browseMenu->Append( type, entryStr );
453 browseMenu->Bind( wxEVT_COMMAND_MENU_SELECTED, &PANEL_DESIGN_BLOCK_LIB_TABLE::browseLibrariesHandler,
454 this, type );
455 }
456
457 Layout();
458
459 m_notebook->Bind( wxEVT_AUINOTEBOOK_PAGE_CLOSE, &PANEL_DESIGN_BLOCK_LIB_TABLE::onNotebookPageCloseRequest, this );
460 m_notebook->Bind( wxEVT_AUINOTEBOOK_PAGE_CHANGING, &PANEL_DESIGN_BLOCK_LIB_TABLE::onNotebookPageChangeRequest, this );
461 // This is the button only press for the browse button instead of the menu
463
464 m_parent->SetCanCloseCheck(
465 [this]()
466 {
467 for( int ii = 0; ii < (int) m_notebook->GetPageCount(); ++ii )
468 {
470 static_cast<LIB_TABLE_NOTEBOOK_PANEL*>( m_notebook->GetPage( ii ) );
471
472 if( panel->GetClosable() )
473 {
474 bool wasDirty = panel->TableModified();
475
476 if( !panel->GetCanClose() )
477 return false;
478
479 if( wasDirty && !panel->TableModified() )
480 {
481 m_parent->m_GlobalTableChanged = true;
482 m_parent->m_ProjectTableChanged = true;
483 }
484 }
485 }
486
487 return true;
488 } );
489}
490
491
493{
494 wxMenu* browseMenu = m_browseButton->GetSplitButtonMenu();
495
496 for( auto& [type, desc] : m_supportedDesignBlockFiles )
497 {
498 browseMenu->Unbind( wxEVT_COMMAND_MENU_SELECTED, &PANEL_DESIGN_BLOCK_LIB_TABLE::browseLibrariesHandler,
499 this, type );
500 }
501
503
504 // Delete the GRID_TRICKS.
505 // (Notebook page GRID_TRICKS are deleted by LIB_TABLE_NOTEBOOK_PANEL.)
506 m_path_subs_grid->PopEventHandler( true );
507}
508
509
511{
513 {
515 {
517 continue;
518 }
519
521
522 if( !pi )
523 continue;
524
525 if( const IO_BASE::IO_FILE_DESC& desc = pi->GetLibraryDesc() )
526 m_supportedDesignBlockFiles.emplace( type, desc );
527 }
528}
529
530
535
536
538{
539 return static_cast<LIB_TABLE_NOTEBOOK_PANEL*>( m_notebook->GetPage( aPage ) )->GetGrid();
540}
541
542
544{
545 for( int page = 0 ; page < (int) m_notebook->GetPageCount(); ++page )
546 {
547 WX_GRID* grid = get_grid( page );
548
550 [&]( int aRow, int aCol )
551 {
552 // show the tabbed panel holding the grid we have flunked:
553 if( m_notebook->GetSelection() != page )
554 m_notebook->SetSelection( page );
555
556 grid->MakeCellVisible( aRow, 0 );
557 grid->SetGridCursor( aRow, aCol );
558 } ) )
559 {
560 return false;
561 }
562 }
563
564 return true;
565}
566
567
572
573
578
579
584
585
590
591
593{
595 aEvent.Veto();
596 else
597 aEvent.Skip();
598}
599
600
602{
603 wxAuiNotebook* notebook = (wxAuiNotebook*) aEvent.GetEventObject();
604 wxWindow* page = notebook->GetPage( aEvent.GetSelection() );
605
606 if( LIB_TABLE_NOTEBOOK_PANEL* panel = dynamic_cast<LIB_TABLE_NOTEBOOK_PANEL*>( page ) )
607 {
608 if( panel->GetClosable() )
609 {
610 if( !panel->GetCanClose() )
611 aEvent.Veto();
612 }
613 else
614 {
615 aEvent.Veto();
616 }
617 }
618}
619
620
621// @todo refactor this function into single location shared with PANEL_SYM_LIB_TABLE
623{
624 if( !cur_grid()->CommitPendingChanges() )
625 return;
626
627 wxArrayInt selectedRows = cur_grid()->GetSelectedRows();
628
629 if( selectedRows.empty() && cur_grid()->GetGridCursorRow() >= 0 )
630 selectedRows.push_back( cur_grid()->GetGridCursorRow() );
631
632 wxArrayInt rowsToMigrate;
634 wxString msg;
635
636 for( int row : selectedRows )
637 {
638 if( cur_grid()->GetCellValue( row, COL_TYPE ) != kicadType )
639 rowsToMigrate.push_back( row );
640 }
641
642 if( rowsToMigrate.size() <= 0 )
643 {
644 wxMessageBox( _( "Select one or more rows containing libraries to save as current KiCad format." ) );
645 return;
646 }
647 else
648 {
649 if( rowsToMigrate.size() == 1 )
650 {
651 msg.Printf( _( "Save '%s' as current KiCad format and replace entry in table?" ),
652 cur_grid()->GetCellValue( rowsToMigrate[0], COL_NICKNAME ) );
653 }
654 else
655 {
656 msg.Printf( _( "Save %d libraries as current KiCad format and replace entries in table?" ),
657 (int) rowsToMigrate.size() );
658 }
659
660 if( !IsOK( m_parent, msg ) )
661 return;
662 }
663
664 for( int row : rowsToMigrate )
665 {
666 wxString relPath = cur_grid()->GetCellValue( row, COL_URI );
667 wxString resolvedPath = ExpandEnvVarSubstitutions( relPath, m_project );
668 wxFileName legacyLib( resolvedPath );
669
670 if( !legacyLib.Exists() )
671 {
672 msg.Printf( _( "Library '%s' not found." ), relPath );
673 DisplayErrorMessage( wxGetTopLevelParent( this ), msg );
674 continue;
675 }
676
677 wxFileName newLib( resolvedPath );
678 newLib.AppendDir( newLib.GetName() + "." + FILEEXT::KiCadDesignBlockLibPathExtension );
679 newLib.SetName( "" );
680 newLib.ClearExt();
681
682 if( newLib.DirExists() )
683 {
684 msg.Printf( _( "Folder '%s' already exists. Do you want overwrite any existing design blocks?" ),
685 newLib.GetFullPath() );
686
687 switch( wxMessageBox( msg, _( "Migrate Library" ), wxYES_NO | wxCANCEL | wxICON_QUESTION, m_parent ) )
688 {
689 case wxYES: break;
690 case wxNO: continue;
691 case wxCANCEL: return;
692 }
693 }
694
695 wxString options = cur_grid()->GetCellValue( row, COL_OPTIONS );
696 std::map<std::string, UTF8> props( LIBRARY_TABLE::ParseOptions( options.ToStdString() ) );
697
698 if( DESIGN_BLOCK_IO_MGR::ConvertLibrary( &props, legacyLib.GetFullPath(), newLib.GetFullPath() ) )
699 {
700 relPath = NormalizePath( newLib.GetFullPath(), &Pgm().GetLocalEnvVariables(), m_project );
701
702 cur_grid()->SetCellValue( row, COL_URI, relPath );
703 cur_grid()->SetCellValue( row, COL_TYPE, kicadType );
704 }
705 else
706 {
707 DisplayErrorMessage( m_parent, wxString::Format( _( "Failed to save design block library file '%s'." ),
708 newLib.GetFullPath() ) );
709 }
710 }
711}
712
713
715{
716 if( !cur_grid()->CommitPendingChanges() )
717 return;
718
720
721 // We are bound both to the menu and button with this one handler
722 // So we must set the file type based on it
723 if( event.GetEventType() == wxEVT_BUTTON )
724 {
725 // Let's default to adding a kicad design block file for just the design block
727 }
728 else
729 {
730 fileType = static_cast<DESIGN_BLOCK_IO_MGR::DESIGN_BLOCK_FILE_T>( event.GetId() );
731 }
732
734 return;
735
738
739 wxString title = wxString::Format( _( "Select %s Library" ), DESIGN_BLOCK_IO_MGR::ShowType( fileType ) );
740 wxString dummy;
741 wxString* lastDir;
742
743 if( m_notebook->GetSelection() == 0 )
744 lastDir = cfg ? &cfg->m_lastDesignBlockLibDir : &dummy;
745 else
746 lastDir = &m_lastProjectLibDir;
747
748 wxArrayString files;
749 wxWindow* topLevelParent = wxGetTopLevelParent( this );
750
751 if( fileDesc.m_IsFile )
752 {
753 wxFileDialog dlg( topLevelParent, title, *lastDir, wxEmptyString, fileDesc.FileFilter(),
754 wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE );
755
757
758 if( dlg.ShowModal() == wxID_CANCEL )
759 return;
760
761 dlg.GetPaths( files );
762 *lastDir = dlg.GetDirectory();
763 }
764 else
765 {
766 wxDirDialog dlg( topLevelParent, title, *lastDir,
767 wxDD_DEFAULT_STYLE | wxDD_DIR_MUST_EXIST | wxDD_MULTIPLE );
768
769 if( dlg.ShowModal() == wxID_CANCEL )
770 return;
771
772 dlg.GetPaths( files );
773
774 if( !files.IsEmpty() )
775 {
776 wxFileName first( files.front() );
777 *lastDir = first.GetPath();
778 }
779 }
780
781 // Drop the last directory if the path is a .pretty folder
783 cfg->m_lastDesignBlockLibDir = cfg->m_lastDesignBlockLibDir.BeforeLast( wxFileName::GetPathSeparator() );
784
785 const ENV_VAR_MAP& envVars = Pgm().GetLocalEnvVariables();
786 bool addDuplicates = false;
787 bool applyToAll = false;
788 wxString warning = _( "Warning: Duplicate Nicknames" );
789 wxString msg = _( "An item nicknamed '%s' already exists." );
790 wxString detailedMsg = _( "One of the nicknames will need to be changed." );
791
792 for( const wxString& filePath : files )
793 {
794 wxFileName fn( filePath );
795 wxString nickname = LIB_ID::FixIllegalChars( fn.GetName(), true );
796 bool doAdd = true;
797
799 nickname = LIB_ID::FixIllegalChars( fn.GetFullName(), true ).wx_str();
800
801 if( cur_model()->ContainsNickname( nickname ) )
802 {
803 if( !applyToAll )
804 {
805 // The cancel button adds the library to the table anyway
806 addDuplicates = OKOrCancelDialog( m_parent, warning, wxString::Format( msg, nickname ), detailedMsg,
807 _( "Skip" ), _( "Add Anyway" ), &applyToAll ) == wxID_CANCEL;
808 }
809
810 doAdd = addDuplicates;
811 }
812
813 if( doAdd && cur_grid()->AppendRows( 1 ) )
814 {
815 int last_row = cur_grid()->GetNumberRows() - 1;
816
817 cur_grid()->SetCellValue( last_row, COL_NICKNAME, nickname );
818 cur_grid()->SetCellValue( last_row, COL_TYPE, DESIGN_BLOCK_IO_MGR::ShowType( fileType ) );
819
820 // try to use path normalized to an environmental variable or project path
821 wxString path = NormalizePath( filePath, &envVars, m_project->GetProjectPath() );
822
823 // Do not use the project path in the global library table. This will almost
824 // assuredly be wrong for a different project.
825 if( m_notebook->GetSelection() == 0 && path.Contains( wxT( "${KIPRJMOD}" ) ) )
826 path = fn.GetFullPath();
827
828 cur_grid()->SetCellValue( last_row, COL_URI, path );
829 }
830 }
831
832 if( !files.IsEmpty() )
833 {
834 cur_grid()->MakeCellVisible( cur_grid()->GetNumberRows() - 1, COL_ENABLED );
835 cur_grid()->SetGridCursor( cur_grid()->GetNumberRows() - 1, COL_NICKNAME );
836 }
837}
838
839
841{
842 // Account for scroll bars
843 aWidth -= ( m_path_subs_grid->GetSize().x - m_path_subs_grid->GetClientSize().x );
844
845 m_path_subs_grid->AutoSizeColumn( 0 );
846 m_path_subs_grid->SetColSize( 0, std::max( 72, m_path_subs_grid->GetColSize( 0 ) ) );
847 m_path_subs_grid->SetColSize( 1, std::max( 120, aWidth - m_path_subs_grid->GetColSize( 0 ) ) );
848}
849
850
852{
853 adjustPathSubsGridColumns( event.GetSize().GetX() );
854
855 event.Skip();
856}
857
858
860{
861 if( !cur_grid()->CommitPendingChanges() )
862 return false;
863
864 if( !verifyTables() )
865 return false;
866
868
869 std::optional<LIBRARY_TABLE*> optTable = manager.Table( LIBRARY_TABLE_TYPE::DESIGN_BLOCK,
871 wxCHECK( optTable.has_value(), false );
872 LIBRARY_TABLE* globalTable = optTable.value();
873
874 if( get_model( 0 )->Table() != *globalTable )
875 {
876 m_parent->m_GlobalTableChanged = true;
877 *globalTable = get_model( 0 )->Table();
878
879 globalTable->Save().map_error(
880 []( const LIBRARY_ERROR& aError )
881 {
882 wxMessageBox( _( "Error saving global library table:\n\n" ) + aError.message,
883 _( "File Save Error" ), wxOK | wxICON_ERROR );
884 } );
885 }
886
888
889 if( optTable.has_value() && get_model( 1 )->Table().Path() == optTable.value()->Path() )
890 {
891 LIBRARY_TABLE* projectTable = optTable.value();
892
893 if( get_model( 1 )->Table() != *projectTable )
894 {
895 m_parent->m_ProjectTableChanged = true;
896 *projectTable = get_model( 1 )->Table();
897
898 projectTable->Save().map_error(
899 []( const LIBRARY_ERROR& aError )
900 {
901 wxMessageBox( _( "Error saving project library table:\n\n" ) + aError.message,
902 _( "File Save Error" ), wxOK | wxICON_ERROR );
903 } );
904 }
905 }
906
908 return true;
909}
910
911
915{
916 wxRegEx re( ".*?(\\$\\{(.+?)\\})|(\\$\\((.+?)\\)).*?", wxRE_ADVANCED );
917 wxASSERT( re.IsValid() ); // wxRE_ADVANCED is required.
918
919 std::set<wxString> unique;
920
921 // clear the table
922 m_path_subs_grid->ClearRows();
923
924 for( int page = 0 ; page < (int) m_notebook->GetPageCount(); ++page )
925 {
927
928 for( int row = 0; row < model->GetNumberRows(); ++row )
929 {
930 wxString uri = model->GetValue( row, COL_URI );
931
932 while( re.Matches( uri ) )
933 {
934 wxString envvar = re.GetMatch( uri, 2 );
935
936 // if not ${...} form then must be $(...)
937 if( envvar.IsEmpty() )
938 envvar = re.GetMatch( uri, 4 );
939
940 // ignore duplicates
941 unique.insert( envvar );
942
943 // delete the last match and search again
944 uri.Replace( re.GetMatch( uri, 0 ), wxEmptyString );
945 }
946 }
947 }
948
949 // Make sure this special environment variable shows up even if it was
950 // not used yet. It is automatically set by KiCad to the directory holding
951 // the current project.
952 unique.insert( PROJECT_VAR_NAME );
954
955 // This special environment variable is used to locate 3d shapes
956 unique.insert( ENV_VAR::GetVersionedEnvVarName( wxS( "3DMODEL_DIR" ) ) );
957
958 for( const wxString& evName : unique )
959 {
960 int row = m_path_subs_grid->GetNumberRows();
961 m_path_subs_grid->AppendRows( 1 );
962
963 m_path_subs_grid->SetCellValue( row, 0, wxT( "${" ) + evName + wxT( "}" ) );
964 m_path_subs_grid->SetCellEditor( row, 0, new GRID_CELL_READONLY_TEXT_EDITOR() );
965
966 wxString evValue;
967 wxGetEnv( evName, &evValue );
968 m_path_subs_grid->SetCellValue( row, 1, evValue );
969 m_path_subs_grid->SetCellEditor( row, 1, new GRID_CELL_READONLY_TEXT_EDITOR() );
970 }
971
972 adjustPathSubsGridColumns( m_path_subs_grid->GetRect().GetWidth() );
973}
974
975//-----</event handlers>---------------------------------
976
977void InvokeEditDesignBlockLibTable( KIWAY* aKiway, wxWindow *aParent )
978{
979 DIALOG_EDIT_LIBRARY_TABLES dlg( aParent, _( "Design Block Libraries" ) );
980
981 dlg.InstallPanel( new PANEL_DESIGN_BLOCK_LIB_TABLE( &dlg, &aKiway->Prj() ) );
982
983 if( dlg.ShowModal() == wxID_CANCEL )
984 return;
985
986 if( dlg.m_GlobalTableChanged )
988
989 if( dlg.m_ProjectTableChanged )
990 {
991 // Trigger a reload of the table and cancel an in-progress background load
993 }
994
995 // Trigger a load of any new block libraries
997
998 std::string payload = "";
999 aKiway->ExpressMail( FRAME_SCH, MAIL_RELOAD_LIB, payload );
1000 aKiway->ExpressMail( FRAME_PCB_EDITOR, MAIL_RELOAD_LIB, payload );
1001
1002 return;
1003}
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition bitmap.cpp:110
void openTable(const LIBRARY_TABLE_ROW &aRow) override
DESIGN_BLOCK_GRID_TRICKS(PANEL_DESIGN_BLOCK_LIB_TABLE *aPanel, WX_GRID *aGrid)
PANEL_DESIGN_BLOCK_LIB_TABLE * m_panel
@ KICAD_SEXP
S-expression KiCad file format.
@ DESIGN_BLOCK_FILE_UNKNOWN
0 is not a legal menu id on Mac
static const wxString ShowType(DESIGN_BLOCK_FILE_T aFileType)
static DESIGN_BLOCK_FILE_T GuessPluginTypeFromLibPath(const wxString &aLibPath, int aCtl=0)
static DESIGN_BLOCK_FILE_T EnumFromStr(const wxString &aFileType)
static bool ConvertLibrary(std::map< std::string, UTF8 > *aOldFileProps, const wxString &aOldFilePath, const wxString &aNewFilePath)
Convert a design block library to the latest KiCad format.
static DESIGN_BLOCK_IO * FindPlugin(DESIGN_BLOCK_FILE_T aFileType)
This class builds a wxGridTableBase by wrapping an #DESIGN_BLOCK_LIB_TABLE object.
void SetValue(int aRow, int aCol, const wxString &aValue) override
wxString getFileTypes(WX_GRID *aGrid, int aRow) override
const std::map< DESIGN_BLOCK_IO_MGR::DESIGN_BLOCK_FILE_T, IO_BASE::IO_FILE_DESC > & m_supportedDesignBlockFiles
DESIGN_BLOCK_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, const std::map< DESIGN_BLOCK_IO_MGR::DESIGN_BLOCK_FILE_T, IO_BASE::IO_FILE_DESC > &aSupportedFiles)
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
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
wxString m_lastDesignBlockLibDir
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 PROJECT & Prj() const
Return the PROJECT associated with this KIWAY.
Definition kiway.cpp:205
The interface used by the classes that actually can load IO plugins for the different parts of KiCad ...
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.
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)
void GetPaths(wxArrayString &aPathArray)
virtual wxDirTraverseResult OnOpenError(const wxString &aOpenErrorName) override
std::unordered_map< wxString, int > m_failedDirs
std::vector< std::string > m_searchExtensions
void GetFailedPaths(wxArrayString &aPathArray)
LIBRARY_TRAVERSER(std::vector< std::string > aSearchExtensions, wxString aInitialDir)
std::unordered_map< wxString, int > m_foundDirs
virtual wxDirTraverseResult OnDir(const wxString &aDirName) override
virtual wxDirTraverseResult OnFile(const wxString &aFileName) override
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)
PANEL_DESIGN_BLOCK_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 moveUpHandler(wxCommandEvent &event) override
void onMigrateLibraries(wxCommandEvent &event) override
void onNotebookPageCloseRequest(wxAuiNotebookEvent &aEvent)
std::vector< std::shared_ptr< LIBRARY_TABLE > > m_nestedTables
void browseLibrariesHandler(wxCommandEvent &event)
void populateEnvironReadOnlyTable()
Populate the readonly environment variable table with names and values by examining all the full_uri ...
std::map< DESIGN_BLOCK_IO_MGR::DESIGN_BLOCK_FILE_T, IO_BASE::IO_FILE_DESC > m_supportedDesignBlockFiles
void moveDownHandler(wxCommandEvent &event) override
DIALOG_EDIT_LIBRARY_TABLES * m_parent
void onNotebookPageChangeRequest(wxAuiNotebookEvent &aEvent)
DESIGN_BLOCK_LIB_TABLE_GRID_DATA_MODEL * get_model(int aPage) const
void appendRowHandler(wxCommandEvent &event) override
PANEL_DESIGN_BLOCK_LIB_TABLE(DIALOG_EDIT_LIBRARY_TABLES *aParent, PROJECT *aProject)
void onSizeGrid(wxSizeEvent &event) override
void deleteRowHandler(wxCommandEvent &event) override
void AddTable(LIBRARY_TABLE *table, const wxString &aTitle, bool aClosable)
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.
DESIGN_BLOCK_LIB_TABLE_GRID_DATA_MODEL * cur_model() const
static wxString GetDefaultUserDesignBlocksPath()
Gets the default path we point users to create projects.
Definition paths.cpp:104
void PreloadDesignBlockLibraries(KIWAY *aKiway)
Starts a background job to preload the global and project design block libraries.
Definition pgm_base.cpp:887
virtual ENV_VAR_MAP & GetLocalEnvVariables() const
Definition pgm_base.cpp:787
virtual LIBRARY_MANAGER & GetLibraryManager() const
Definition pgm_base.h:132
Container for project specific data.
Definition project.h:66
wxString wx_str() const
Definition utf8.cpp:45
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)
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_PCB_EDITOR
Definition frame_type.h:42
@ FRAME_SCH
Definition frame_type.h:34
static const std::string KiCadDesignBlockLibPathExtension
static const std::string DesignBlockLibraryTableFileName
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 InvokeEditDesignBlockLibTable(KIWAY *aKiway, wxWindow *aParent)
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
wxString message
Container that describes file type info for the add a library options.
bool m_IsFile
Whether the library is a folder or a file.
wxString m_Description
Description shown in the file picker dialog.
wxString m_FileFilter
Filter used for file pickers if m_IsFile is true.
DESIGN_BLOCK_IO_MGR::DESIGN_BLOCK_FILE_T m_Plugin
wxString m_FolderSearchExtension
In case of folders it stands for extensions of files stored inside.
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.