56#include <wx/filedlg.h>
92 wxString* aMRUDirectory,
const wxString& aProjectPath ) :
98 void SetValue(
int aRow,
int aCol,
const wxString &aValue )
override
100 wxCHECK( aRow < (
int)
size(), );
111 if( pluginType != SCH_IO_MGR::SCH_FILE_UNKNOWN )
124 wxString
filter =
_(
"Symbol Library Tables" );
129 filter << wxString::Format(
_(
" (%s)|%s" ), wxFileSelectorDefaultWildcardStr,
130 wxFileSelectorDefaultWildcardStr );
146 return wxEmptyString;
155 std::function<
void( wxCommandEvent& )> aAddHandler ) :
175 const wxString& options = row.
Options();
176 wxString
result = options;
177 std::map<std::string, UTF8> choices;
181 pi->GetLibraryOptions( &choices );
201 wxMessageBox(
_(
"Unable to load library table." ) );
211 return wxT(
"(sym_lib_table" );
226 for(
int ii = 2; ii < (int)
m_notebook->GetPageCount(); ++ii )
232 for(
int attempts = 0; attempts < 3; ++attempts )
240 AddTable( aTable.get(), aTitle,
true );
244 for(
int attempts = 0; attempts < 3; ++attempts )
252 wxString projectPath =
m_project->GetProjectPath();
258 if(
table->Path().StartsWith( projectPath ) )
266 wxString* lastGlobalLibDir =
nullptr;
270 if( cfg->m_lastSymbolLibDir.IsEmpty() )
273 lastGlobalLibDir = &cfg->m_lastSymbolLibDir;
277 lastGlobalLibDir, wxEmptyString ),
295 [
this]( wxCommandEvent& event )
303 int prevWidth =
grid->GetColSize( aCol );
305 grid->AutoSizeColumn( aCol,
false );
306 grid->SetColSize( aCol, std::max( prevWidth,
grid->GetColSize( aCol ) ) );
315 if(
grid->GetNumberRows() > 0 )
318 grid->SelectRow( 0 );
333 for(
const SCH_IO_MGR::SCH_FILE_T& type : SCH_IO_MGR::SCH_FILE_T_vector )
335 if( type == SCH_IO_MGR::SCH_NESTED_TABLE )
349 wxASSERT(
table.has_value() );
356 if( projectTable.has_value() )
357 AddTable( projectTable.value(),
_(
"Project Specific Libraries" ),
false );
386 auto joinExtensions =
387 [](
const std::vector<std::string>& aExts ) -> wxString
391 for(
const std::string& ext : aExts )
396 result << wxT(
"." ) << ext;
406 if( !desc.m_FileExtensions.empty() )
407 entryStr << wxString::Format( wxS(
" (%s)" ), joinExtensions( desc.m_FileExtensions ) );
414 if( type == SCH_IO_MGR::SCH_KICAD )
417 folderEntry << wxString::Format( wxS(
" (%s)" ),
_(
"folder with .kicad_sym files" ) );
433 for(
int ii = 0; ii < (int)
m_notebook->GetPageCount(); ++ii )
447 m_parent->m_GlobalTableChanged =
true;
448 m_parent->m_ProjectTableChanged =
true;
477 for(
const SCH_IO_MGR::SCH_FILE_T& type : SCH_IO_MGR::SCH_FILE_T_vector )
486 if( !desc.m_FileExtensions.empty() )
519 for(
int page = 0 ; page < (int)
m_notebook->GetPageCount(); ++page )
524 [&](
int aRow,
int aCol )
530 grid->MakeCellVisible( aRow, 0 );
531 grid->SetGridCursor( aRow, aCol );
544 if( !
cur_grid()->CommitPendingChanges() )
547 SCH_IO_MGR::SCH_FILE_T
fileType = SCH_IO_MGR::SCH_FILE_UNKNOWN;
548 bool selectingFolder =
false;
551 if( event.GetEventType() == wxEVT_BUTTON )
560 selectingFolder =
true;
567 if(
fileType == SCH_IO_MGR::SCH_FILE_UNKNOWN )
582 wxWindow* topLevelParent = wxGetTopLevelParent(
this );
585 if( selectingFolder )
587 wxDirDialog dlg( topLevelParent, title, *lastDir, wxDD_DEFAULT_STYLE | wxDD_DIR_MUST_EXIST | wxDD_MULTIPLE );
589 if( dlg.ShowModal() == wxID_CANCEL )
592 dlg.GetPaths( files );
594 if( !files.IsEmpty() )
596 wxFileName first( files.front() );
597 *lastDir = first.GetPath();
609 wxFileDialog dlg( topLevelParent, title, *lastDir, wxEmptyString, fileDesc.
FileFilter(),
610 wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE );
614 if( dlg.ShowModal() == wxID_CANCEL )
617 dlg.GetPaths( files );
618 *lastDir = dlg.GetDirectory();
621 bool addDuplicates =
false;
622 bool applyToAll =
false;
623 wxString warning =
_(
"Warning: Duplicate Nicknames" );
624 wxString msg =
_(
"An item nicknamed '%s' already exists." );
625 wxString detailedMsg =
_(
"One of the nicknames will need to be changed." );
627 for(
const wxString& filePath : files )
629 wxFileName fn( filePath );
633 if(
cur_model()->ContainsNickname( nickname ) )
638 wxString::Format( msg, nickname ), detailedMsg,
639 _(
"Skip" ),
_(
"Add Anyway" ),
640 &applyToAll ) == wxID_CANCEL;
643 doAdd = addDuplicates;
646 if( doAdd &&
cur_grid()->AppendRows( 1 ) )
648 int last_row =
cur_grid()->GetNumberRows() - 1;
658 if(
m_notebook->GetSelection() == 0 &&
path.Contains( wxT(
"${KIPRJMOD}" ) ) )
659 path = fn.GetFullPath();
665 if( !files.IsEmpty() )
699 if( !
cur_grid()->CommitPendingChanges() )
705 if(
grid->GetNumberRows() > 0 && !
IsOK(
this, wxString::Format(
_(
"This action will reset your global library "
706 "table on disk and cannot be undone." ) ) ) )
711 wxString* lastGlobalLibDir =
nullptr;
715 if( cfg->m_lastSymbolLibDir.IsEmpty() )
718 lastGlobalLibDir = &cfg->m_lastSymbolLibDir;
727 schface->PreloadLibraries( &
m_parent->Kiway() );
731 wxGridTableBase*
table =
grid->GetTable();
736 wxASSERT( newTable );
741 lastGlobalLibDir, wxEmptyString ),
750 m_parent->m_GlobalTableChanged =
true;
754 if(
grid->GetNumberRows() > 0 )
757 grid->SelectRow( 0 );
779 wxAuiNotebook* notebook = (wxAuiNotebook*) aEvent.GetEventObject();
780 wxWindow* page = notebook->GetPage( aEvent.GetSelection() );
784 if( panel->GetClosable() )
786 if( !panel->GetCanClose() )
799 if( !
cur_grid()->CommitPendingChanges() )
802 wxArrayInt selectedRows =
cur_grid()->GetSelectedRows();
804 if( selectedRows.empty() &&
cur_grid()->GetGridCursorRow() >= 0 )
805 selectedRows.push_back(
cur_grid()->GetGridCursorRow() );
807 wxArrayInt legacyRows;
812 for(
int row : selectedRows )
817 legacyRows.push_back( row );
821 if( legacyRows.size() <= 0 )
823 wxMessageBox(
_(
"Select one or more rows containing libraries to save as current KiCad format." ) );
828 if( legacyRows.size() == 1 )
830 msg.Printf(
_(
"Save '%s' as current KiCad format (*.kicad_sym) and replace legacy entry in table?" ),
835 msg.Printf(
_(
"Save %d libraries as current KiCad format (*.kicad_sym) and replace legacy entries "
837 (
int) legacyRows.size() );
844 for(
int row : legacyRows )
848 wxFileName legacyLib( resolvedPath );
850 if( !legacyLib.Exists() )
856 wxFileName newLib( resolvedPath );
857 newLib.SetExt(
"kicad_sym" );
859 if( newLib.Exists() )
861 msg.Printf(
_(
"File '%s' already exists. Do you want overwrite this file?" ), newLib.GetFullPath() );
863 switch( wxMessageBox( msg,
_(
"Migrate Library" ), wxYES_NO | wxCANCEL | wxICON_QUESTION,
m_parent ) )
867 case wxCANCEL:
return;
885 newLib.GetFullPath() ) );
893 if( !
cur_grid()->CommitPendingChanges() )
901 wxCHECK( optTable,
false );
904 if(
get_model( 0 )->Table() != *globalTable )
906 m_parent->m_GlobalTableChanged =
true;
909 globalTable->
Save().map_error(
912 wxMessageBox(
_(
"Error saving global library table:\n\n" ) + aError.
message,
913 _(
"File Save Error" ), wxOK | wxICON_ERROR );
919 if( optTable.has_value() &&
get_model( 1 )->Table().Path() == optTable.value()->Path() )
923 if(
get_model( 1 )->Table() != *projectTable )
925 m_parent->m_ProjectTableChanged =
true;
928 projectTable->
Save().map_error(
931 wxMessageBox(
_(
"Error saving project library table:\n\n" ) + aError.
message,
932 _(
"File Save Error" ), wxOK | wxICON_ERROR );
944 wxRegEx re(
".*?(\\$\\{(.+?)\\})|(\\$\\((.+?)\\)).*?", wxRE_ADVANCED );
945 wxASSERT( re.IsValid() );
947 std::set< wxString > unique;
952 for(
int page = 0 ; page < (int)
m_notebook->GetPageCount(); ++page )
956 for(
int row = 0; row <
model->GetNumberRows(); ++row )
960 while( re.Matches( uri ) )
962 wxString envvar = re.GetMatch( uri, 2 );
965 if( envvar.IsEmpty() )
966 envvar = re.GetMatch( uri, 4 );
969 unique.insert( envvar );
972 uri.Replace( re.GetMatch( uri, 0 ), wxEmptyString );
983 for(
const wxString& evName : unique )
988 m_path_subs_grid->SetCellValue( row, 0, wxT(
"${" ) + evName + wxT(
"}" ) );
992 wxGetEnv( evName, &evValue );
1029 if( symbolEditor && symbolEditor->IsSyncLibrariesInProgress() )
1031 symbolEditor->CallAfter( [aKiway, aParent]()
1041 symbolEditor->FreezeLibraryTree();
1043 if( symbolEditor->HasLibModifications() )
1045 msg =
_(
"Modifications have been made to one or more symbol libraries.\n"
1046 "Changes must be saved or discarded before the symbol library table can be modified." );
1050 case wxID_YES: symbolEditor->SaveAll();
break;
1051 case wxID_NO: symbolEditor->RevertAll();
break;
1053 case wxID_CANCEL: symbolEditor->ThawLibraryTree();
return;
1066 symbolEditor->ThawLibraryTree();
1082 schface->PreloadLibraries( aKiway );
1085 symbolEditor->ThawLibraryTree();
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
bool m_ProjectTableChanged
void InstallPanel(wxPanel *aPanel)
bool m_GlobalTableChanged
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...
wxString m_lastSymbolLibDir
Add mouse and command handling (such as cut, copy, and paste) to a WX_GRID instance.
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.
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...
virtual KIWAY_PLAYER * Player(FRAME_T aFrameType, bool doCreate=true, wxTopLevelWindow *aParent=nullptr)
Return the KIWAY_PLAYER* given a FRAME_T.
virtual KIFACE * KiFACE(FACE_T aFaceId, bool doLoad=true)
Return the KIFACE* given a FACE_T.
virtual PROJECT & Prj() const
Return the PROJECT associated with this KIWAY.
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...
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)
static UTF8 FixIllegalChars(const UTF8 &aLibItemName, bool aLib)
Replace illegal LIB_ID item name characters with underscores '_'.
This abstract base class mixes any object derived from #LIB_TABLE into wxGridTableBase so the result ...
virtual LIBRARY_TABLE_ROW & at(size_t aIndex)
int GetNumberRows() override
virtual size_t size() const
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)
bool GetCanClose() override
PANEL_SYM_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)
STD_BITMAP_BUTTON * m_move_up_button
wxAuiNotebook * m_notebook
STD_BITMAP_BUTTON * m_delete_button
SPLIT_BUTTON * m_browseButton
STD_BITMAP_BUTTON * m_move_down_button
STD_BITMAP_BUTTON * m_append_button
WX_GRID * m_path_subs_grid
Dialog to show and edit symbol library tables.
void moveUpHandler(wxCommandEvent &event) override
SYMBOL_LIB_TABLE_GRID_DATA_MODEL * get_model(int aPage) const
virtual ~PANEL_SYM_LIB_TABLE()
void onReset(wxCommandEvent &event) override
bool m_suppressNotebookPageEvents
void deleteRowHandler(wxCommandEvent &event) override
WX_GRID * get_grid(int aPage) const
wxString m_lastProjectLibDir
void onNotebookPageCloseRequest(wxAuiNotebookEvent &aEvent)
void onPageChange(wxAuiNotebookEvent &event) override
void browseLibrariesHandler(wxCommandEvent &event) override
void AddTable(LIBRARY_TABLE *table, const wxString &aTitle, bool aClosable)
void populatePluginList()
bool verifyTables()
Trim important fields, removes blank row entries, and checks for duplicates.
void adjustPathSubsGridColumns(int aWidth)
void OpenTable(const std::shared_ptr< LIBRARY_TABLE > &table, const wxString &aTitle)
void onSizeGrid(wxSizeEvent &event) override
std::vector< std::shared_ptr< LIBRARY_TABLE > > m_nestedTables
void onConvertLegacyLibraries(wxCommandEvent &event) override
void onNotebookPageChangeRequest(wxAuiNotebookEvent &aEvent)
std::map< SCH_IO_MGR::SCH_FILE_T, IO_BASE::IO_FILE_DESC > m_supportedSymFiles
bool TransferDataToWindow() override
wxArrayString m_pluginChoices
void moveDownHandler(wxCommandEvent &event) override
PANEL_SYM_LIB_TABLE(DIALOG_EDIT_LIBRARY_TABLES *aParent, PROJECT *m_project)
SYMBOL_LIB_TABLE_GRID_DATA_MODEL * cur_model() const
void populateEnvironReadOnlyTable()
Populate the readonly environment variable table with names and values by examining all the full_uri ...
WX_GRID * cur_grid() const
DIALOG_EDIT_LIBRARY_TABLES * m_parent
bool TransferDataFromWindow() override
void appendRowHandler(wxCommandEvent &event) override
static wxString GetDefaultUserSymbolsPath()
Gets the default path we point users to create projects.
virtual ENV_VAR_MAP & GetLocalEnvVariables() const
virtual LIBRARY_MANAGER & GetLibraryManager() const
static SYMBOL_LIBRARY_ADAPTER * SymbolLibAdapter(PROJECT *aProject)
Accessor for project symbol library manager adapter.
Container for project specific data.
static bool ConvertLibrary(std::map< std::string, UTF8 > *aOldFileProps, const wxString &aOldFilePath, const wxString &aNewFilepath)
Convert a schematic symbol library to the latest KiCad format.
static SCH_FILE_T EnumFromStr(const wxString &aFileType)
Return the #SCH_FILE_T from the corresponding plugin type name: "kicad", "legacy",...
static const wxString ShowType(SCH_FILE_T aFileType)
Return a brief name for a plugin, given aFileType enum.
static SCH_FILE_T GuessPluginTypeFromLibPath(const wxString &aLibPath, int aCtl=0)
Return a plugin type given a symbol library using the file extension of aLibPath.
The symbol library editor main window.
SYMBOL_GRID_TRICKS(PANEL_SYM_LIB_TABLE *aPanel, WX_GRID *aGrid, std::function< void(wxCommandEvent &)> aAddHandler)
bool supportsVisibilityColumn() override
wxString getTablePreamble() override
PANEL_SYM_LIB_TABLE * m_panel
void openTable(const LIBRARY_TABLE_ROW &aRow) override
void optionsEditor(int aRow) override
static bool SupportsVisibilityColumn()
An interface to the global shared library manager that is schematic-specific and linked to one projec...
SYMBOL_LIB_TABLE_GRID_DATA_MODEL(DIALOG_SHIM *aParent, WX_GRID *aGrid, const LIBRARY_TABLE &aTableToEdit, SYMBOL_LIBRARY_ADAPTER *aAdapter, const wxArrayString &aPluginChoices, wxString *aMRUDirectory, const wxString &aProjectPath)
void SetValue(int aRow, int aCol, const wxString &aValue) override
wxString getFileTypes(WX_GRID *aGrid, int aRow) override
const wxString ExpandEnvVarSubstitutions(const wxString &aString, const PROJECT *aProject)
Replace any environment variable & text variable references with their values.
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.
bool IsOK(wxWindow *aParent, const wxString &aMessage)
Display a yes/no dialog with aMessage and returns the user response.
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
int UnsavedChangesDialog(wxWindow *parent, const wxString &aMessage, bool *aApplyToAll)
A specialized version of HandleUnsavedChanges which handles an apply-to-all checkbox.
This file is part of the common library.
wxString NormalizePath(const wxFileName &aFilePath, const ENV_VAR_MAP *aEnvVars, const wxString &aProjectPath)
Normalize a file path to an environmental variable, if possible.
Helper functions to substitute paths with environmental variables.
Functions related to environment variables, including help functions.
@ FRAME_SCH_SYMBOL_EDITOR
static const std::string SymbolLibraryTableFileName
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.
KICOMMON_API wxString GetVersionedEnvVarName(const wxString &aBaseName)
Construct a versioned environment variable based on this KiCad major version.
static constexpr int FIRST_MENU_ID
Special menu ID for folder-based KiCad symbol library format.
static constexpr int ID_PANEL_SYM_LIB_KICAD_FOLDER
void InvokeSchEditSymbolLibTable(KIWAY *aKiway, wxWindow *aParent)
PGM_BASE & Pgm()
The global program "get" accessor.
#define PROJECT_VAR_NAME
A variable name whose value holds the current project directory.
T * GetAppSettings(const char *aFilename)
std::vector< FAB_LAYER_COLOR > dummy
MODEL3D_FORMAT_TYPE fileType(const char *aFileName)
Container that describes file type info.
bool m_IsFile
Whether the library is a folder or a file.
wxString FileFilter() const
Implement a participant in the KIWAY alchemy.
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::vector< std::vector< std::string > > table
wxString result
Test unit parsing edge cases and error handling.
Definition of file extensions used in Kicad.