KiCad PCB EDA Suite
Loading...
Searching...
No Matches
design_block_pane.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24#include <design_block.h>
26#include <design_block_io.h>
27#include <paths.h>
28#include <env_paths.h>
29#include <pgm_base.h>
30#include <common.h>
31#include <kidialog.h>
36#include <kiface_base.h>
37#include <core/kicad_algo.h>
38#include <template_fieldnames.h>
39#include <wx/sizer.h>
40#include <wx/textdlg.h>
41#include <confirm.h>
43#include <tool/tool_manager.h>
44
46 std::vector<LIB_ID>& aHistoryList ) :
47 WX_PANEL( aParent ),
48 m_frame( aParent )
49{
50 m_frame->Bind( wxEVT_AUI_PANE_CLOSE, &DESIGN_BLOCK_PANE::OnClosed, this );
51 m_frame->Bind( EDA_LANG_CHANGED, &DESIGN_BLOCK_PANE::OnLanguageChanged, this );
52}
53
54
56{
57 m_frame->Unbind( wxEVT_AUI_PANE_CLOSE, &DESIGN_BLOCK_PANE::OnClosed, this );
58 m_frame->Unbind( EDA_LANG_CHANGED, &DESIGN_BLOCK_PANE::OnLanguageChanged, this );
59}
60
61
62void DESIGN_BLOCK_PANE::OnLanguageChanged( wxCommandEvent& aEvent )
63{
64 if( m_chooserPanel )
65 m_chooserPanel->ShowChangedLanguage();
66
68
69 aEvent.Skip();
70}
71
72
73void DESIGN_BLOCK_PANE::OnClosed( wxAuiManagerEvent& aEvent )
74{
75 if( APP_SETTINGS_BASE* cfg = m_frame->config() )
76 {
77 if( IsShownOnScreen() ) // Ensure the panel is shown when trying to save its size
78 m_frame->SaveSettings( cfg );
79 }
80
81 aEvent.Skip();
82}
83
84
86{
87 m_chooserPanel->SaveSettings();
88}
89
90
92{
93 return m_chooserPanel->GetSelectedLibId( aUnit );
94}
95
96
98{
99 m_chooserPanel->SelectLibId( aLibId );
100}
101
102
104{
105 m_chooserPanel->RefreshLibs();
106}
107
108
109DESIGN_BLOCK* DESIGN_BLOCK_PANE::GetDesignBlock( const LIB_ID& aLibId, bool aUseCacheLib, bool aShowErrorMsg )
110{
111 DESIGN_BLOCK_LIBRARY_ADAPTER* prjLibs = m_frame->Prj().DesignBlockLibs();
112
113 wxCHECK_MSG( prjLibs, nullptr, wxS( "Invalid design block library table." ) );
114
115 DESIGN_BLOCK* designBlock = nullptr;
116
117 try
118 {
119 designBlock = prjLibs->DesignBlockLoadWithOptionalNickname( aLibId, true );
120 }
121 catch( const IO_ERROR& ioe )
122 {
123 if( aShowErrorMsg )
124 {
125 wxString msg = wxString::Format( _( "Error loading design block %s from library '%s'." ),
126 aLibId.GetLibItemName().wx_str(), aLibId.GetLibNickname().wx_str() );
127 DisplayErrorMessage( m_frame, msg, ioe.What() );
128 }
129 }
130
131 return designBlock;
132}
133
134
135DESIGN_BLOCK* DESIGN_BLOCK_PANE::GetSelectedDesignBlock( bool aUseCacheLib, bool aShowErrorMsg )
136{
137 if( !GetSelectedLibId().IsValid() )
138 return nullptr;
139
140 return GetDesignBlock( GetSelectedLibId(), aUseCacheLib, aShowErrorMsg );
141}
142
143
144wxString DESIGN_BLOCK_PANE::CreateNewDesignBlockLibrary( const wxString& aDialogTitle )
145{
146 return createNewDesignBlockLibrary( aDialogTitle );
147}
148
149
150wxString DESIGN_BLOCK_PANE::createNewDesignBlockLibrary( const wxString& aDialogTitle )
151{
152 wxFileName fn;
153 static bool isGlobal = true;
154 FILEDLG_HOOK_NEW_LIBRARY tableChooser( isGlobal );
155
156 if( !m_frame->LibraryFileBrowser( aDialogTitle, false, fn, FILEEXT::KiCadDesignBlockLibPathWildcard(),
157 FILEEXT::KiCadDesignBlockLibPathExtension, false, &tableChooser ) )
158 {
159 return wxEmptyString;
160 }
161
162 isGlobal = tableChooser.GetUseGlobalTable();
163
164 wxString libPath = fn.GetFullPath();
165
167
168 // We can save libs only using DESIGN_BLOCK_IO_MGR::KICAD_SEXP format (.pretty libraries)
170
171 try
172 {
174
175 bool writable = false;
176 bool exists = false;
177
178 try
179 {
180 writable = pi->IsLibraryWritable( libPath );
181 exists = fn.Exists();
182 }
183 catch( const IO_ERROR& )
184 {
185 // best efforts....
186 }
187
188 if( exists )
189 {
190 if( !writable )
191 {
192 wxString msg = wxString::Format( _( "Library %s is read only." ), libPath );
193 m_frame->ShowInfoBarError( msg );
194 return wxEmptyString;
195 }
196 else
197 {
198 wxString msg = wxString::Format( _( "Library %s already exists." ), libPath );
199 KIDIALOG dlg( m_frame, msg, _( "Confirmation" ), wxOK | wxCANCEL | wxICON_WARNING );
200 dlg.SetOKLabel( _( "Overwrite" ) );
201 dlg.DoNotShowCheckbox( __FILE__, __LINE__ );
202
203 if( dlg.ShowModal() == wxID_CANCEL )
204 return wxEmptyString;
205
206 pi->DeleteLibrary( libPath );
207 }
208 }
209
210 pi->CreateLibrary( libPath );
211 }
212 catch( const IO_ERROR& ioe )
213 {
214 DisplayError( m_frame, ioe.What() );
215 return wxEmptyString;
216 }
217
218 AddDesignBlockLibrary( aDialogTitle, libPath, scope );
219
220 return libPath;
221}
222
223
224bool DESIGN_BLOCK_PANE::AddDesignBlockLibrary( const wxString& aDialogTitle, const wxString& aFilename,
225 LIBRARY_TABLE_SCOPE aScope )
226{
227 DESIGN_BLOCK_LIBRARY_ADAPTER* adapter = m_frame->Prj().DesignBlockLibs();
229
230 // TODO(JE) library tables -- figure out where Jeff's added aDialogTitle should be used?
231
232 wxFileName fn( aFilename );
233 wxString libPath = fn.GetFullPath();
234 wxString libName = fn.GetName();
235
236 if( libName.IsEmpty() )
237 return false;
238
239 // Open a dialog to ask for a description
240 wxString description = wxGetTextFromUser( _( "Enter a description for the library:" ), _( "Library Description" ),
241 wxEmptyString, m_frame );
242
244
247
248 wxString type = DESIGN_BLOCK_IO_MGR::ShowType( lib_type );
249
250 // KiCad lib is our default guess. So it might not have the .kicad_blocks extension
251 // In this case, the extension is part of the library name
253 libName = fn.GetFullName();
254
255 // try to use path normalized to an environmental variable or project path
256 wxString normalizedPath = NormalizePath( libPath, &Pgm().GetLocalEnvVariables(), &m_frame->Prj() );
257
258 std::optional<LIBRARY_TABLE*> optTable = manager.Table( LIBRARY_TABLE_TYPE::DESIGN_BLOCK, aScope );
259 wxCHECK( optTable.has_value(), false );
260 LIBRARY_TABLE* table = optTable.value();
261
262 bool success = true;
263 LIBRARY_TABLE_ROW& newRow = table->InsertRow();
264
265 newRow.SetNickname( libName );
266 newRow.SetURI( normalizedPath );
267 newRow.SetType( type );
268 newRow.SetDescription( description );
269
270 table->Save().map_error(
271 [&]( const LIBRARY_ERROR& aError )
272 {
273 DisplayError( m_frame, _( "Error saving library table:\n\n" ) + aError.message );
274 success = false;
275 } );
276
277 if( success )
278 {
279 manager.ReloadTables( aScope, { LIBRARY_TABLE_TYPE::DESIGN_BLOCK } );
280 adapter->LoadOne( libName );
281
282 LIB_ID libID( libName, wxEmptyString );
283 RefreshLibs();
284 SelectLibId( libID );
285 }
286
287 return true;
288}
289
290
291bool DESIGN_BLOCK_PANE::DeleteDesignBlockLibrary( const wxString& aLibName, bool aConfirm )
292{
293 if( aLibName.IsEmpty() )
294 {
295 DisplayErrorMessage( m_frame, _( "Please select a library to delete." ) );
296 return false;
297 }
298
299 if( !m_frame->Prj().DesignBlockLibs()->IsDesignBlockLibWritable( aLibName ) )
300 {
301 wxString msg = wxString::Format( _( "Library '%s' is read only." ), aLibName );
302 m_frame->ShowInfoBarError( msg );
303 return false;
304 }
305
306 // Confirmation
307 wxString msg = wxString::Format( _( "Delete design block library '%s' from disk? This will "
308 "delete all design blocks within the library." ),
309 aLibName.GetData() );
310
311 if( aConfirm && !IsOK( m_frame, msg ) )
312 return false;
313
314 try
315 {
316 m_frame->Prj().DesignBlockLibs()->DeleteLibrary( aLibName );
317 }
318 catch( const IO_ERROR& ioe )
319 {
320 DisplayError( m_frame, ioe.What() );
321 return false;
322 }
323
324 msg.Printf( _( "Design block library '%s' deleted" ), aLibName.GetData() );
325 m_frame->SetStatusText( msg );
326
327 RefreshLibs();
328
329 return true;
330}
331
332
333bool DESIGN_BLOCK_PANE::DeleteDesignBlockFromLibrary( const LIB_ID& aLibId, bool aConfirm )
334{
335 if( !aLibId.IsValid() )
336 return false;
337
338 wxString libname = aLibId.GetLibNickname();
339 wxString dbname = aLibId.GetLibItemName();
340
341 if( !m_frame->Prj().DesignBlockLibs()->IsDesignBlockLibWritable( libname ) )
342 {
343 wxString msg = wxString::Format( _( "Library '%s' is read only." ), libname );
344 m_frame->ShowInfoBarError( msg );
345 return false;
346 }
347
348 // Confirmation
349 wxString msg = wxString::Format( _( "Delete design block '%s' in library '%s' from disk?" ),
350 dbname.GetData(), libname.GetData() );
351
352 if( aConfirm && !IsOK( m_frame, msg ) )
353 return false;
354
355 try
356 {
357 m_frame->Prj().DesignBlockLibs()->DeleteDesignBlock( libname, dbname );
358 }
359 catch( const IO_ERROR& ioe )
360 {
361 DisplayError( m_frame, ioe.What() );
362 return false;
363 }
364
365 msg.Printf( _( "Design block '%s' deleted from library '%s'" ), dbname.GetData(), libname.GetData() );
366
367 m_frame->SetStatusText( msg );
368
369 RefreshLibs();
370
371 return true;
372}
373
374
376{
377 if( !aLibId.IsValid() )
378 return false;
379
380 wxString libname = aLibId.GetLibNickname();
381
382 if( !m_frame->Prj().DesignBlockLibs()->IsDesignBlockLibWritable( libname ) )
383 {
384 wxString msg = wxString::Format( _( "Library '%s' is read only." ), libname );
385 m_frame->ShowInfoBarError( msg );
386 return false;
387 }
388
389 std::unique_ptr<DESIGN_BLOCK> designBlock( GetDesignBlock( aLibId, true, true ) );
390
391 if( !designBlock )
392 return false;
393
394 wxString originalName = designBlock->GetLibId().GetLibItemName();
395 DIALOG_DESIGN_BLOCK_PROPERTIES dlg( m_frame, designBlock.get() );
396
397 if( dlg.ShowModal() != wxID_OK )
398 return false;
399
400 wxString newName = designBlock->GetLibId().GetLibItemName();
401
402 try
403 {
404 if( originalName != newName )
405 {
406 if( m_frame->Prj().DesignBlockLibs()->DesignBlockExists( libname, newName ) )
407 {
408 if( !checkOverwrite( m_frame, libname, newName ) )
409 return false;
410 }
411
412 m_frame->Prj().DesignBlockLibs()->SaveDesignBlock( libname, designBlock.get() );
413 m_frame->Prj().DesignBlockLibs()->DeleteDesignBlock( libname, originalName );
414 }
415 else
416 {
417 m_frame->Prj().DesignBlockLibs()->SaveDesignBlock( libname, designBlock.get() );
418 }
419 }
420 catch( const IO_ERROR& ioe )
421 {
422 DisplayError( m_frame, ioe.What() );
423 return false;
424 }
425
426 RefreshLibs();
427 SelectLibId( designBlock->GetLibId() );
428
429 return true;
430}
431
432
433bool DESIGN_BLOCK_PANE::checkOverwrite( wxWindow* aFrame, wxString& libname, wxString& newName )
434{
435 wxString msg = wxString::Format( _( "Design block '%s' already exists in library '%s'." ),
436 newName.GetData(), libname.GetData() );
437
438 if( OKOrCancelDialog( aFrame, _( "Confirmation" ), msg, _( "Overwrite existing design block?" ), _( "Overwrite" ) )
439 != wxID_OK )
440 {
441 return false;
442 }
443
444 return true;
445}
APP_SETTINGS_BASE is a settings class that should be derived for each standalone KiCad application.
@ KICAD_SEXP
S-expression KiCad file format.
static const wxString ShowType(DESIGN_BLOCK_FILE_T aFileType)
static DESIGN_BLOCK_FILE_T GuessPluginTypeFromLibPath(const wxString &aLibPath, int aCtl=0)
static DESIGN_BLOCK_IO * FindPlugin(DESIGN_BLOCK_FILE_T aFileType)
DESIGN_BLOCK * DesignBlockLoadWithOptionalNickname(const LIB_ID &aDesignBlockId, bool aKeepUUID=false)
Load a design block having aDesignBlockId with possibly an empty nickname.
std::optional< LIB_STATUS > LoadOne(LIB_DATA *aLib) override
Loads or reloads the given library, if it exists.
bool DeleteDesignBlockLibrary(const wxString &aLibName, bool aConfirm)
bool checkOverwrite(wxWindow *aFrame, wxString &libname, wxString &newName)
bool DeleteDesignBlockFromLibrary(const LIB_ID &aLibId, bool aConfirm)
void SelectLibId(const LIB_ID &aLibId)
LIB_ID GetSelectedLibId(int *aUnit=nullptr) const
DESIGN_BLOCK_PANE(EDA_DRAW_FRAME *aParent, const LIB_ID *aPreselect, std::vector< LIB_ID > &aHistoryList)
DESIGN_BLOCK * GetDesignBlock(const LIB_ID &aLibId, bool aUseCacheLib, bool aShowErrorMsg)
Load design block from design block library table.
wxString createNewDesignBlockLibrary(const wxString &aDialogTitle)
PANEL_DESIGN_BLOCK_CHOOSER * m_chooserPanel
bool EditDesignBlockProperties(const LIB_ID &aLibId)
virtual void setLabelsAndTooltips()=0
wxString CreateNewDesignBlockLibrary(const wxString &aDialogTitle)
Creates a new design block library.
virtual void OnLanguageChanged(wxCommandEvent &aEvent)
EDA_DRAW_FRAME * m_frame
DESIGN_BLOCK * GetSelectedDesignBlock(bool aUseCacheLib, bool aShowErrorMsg)
void OnClosed(wxAuiManagerEvent &aEvent)
bool AddDesignBlockLibrary(const wxString &aDialogTitle, const wxString &aFilename, LIBRARY_TABLE_SCOPE aScope)
Add an existing library to a library table (presumed to be either the global or project design block ...
int ShowModal() override
The base class for create windows for drawing purpose.
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
virtual const wxString What() const
A composite of Problem() and Where()
Helper class to create more flexible dialogs, including 'do not show again' checkbox handling.
Definition kidialog.h:42
void DoNotShowCheckbox(wxString file, int line)
Shows the 'do not show again' checkbox.
Definition kidialog.cpp:55
int ShowModal() override
Definition kidialog.cpp:93
void ReloadTables(LIBRARY_TABLE_SCOPE aScope, std::initializer_list< LIBRARY_TABLE_TYPE > aTablesToLoad={})
std::optional< LIBRARY_TABLE * > Table(LIBRARY_TABLE_TYPE aType, LIBRARY_TABLE_SCOPE aScope)
Retrieves a given table; creating a new empty project table if a valid project is loaded and the give...
void SetNickname(const wxString &aNickname)
void SetType(const wxString &aType)
void SetDescription(const wxString &aDescription)
void SetURI(const wxString &aUri)
A logical library item identifier and consists of various portions much like a URI.
Definition lib_id.h:49
bool IsValid() const
Check if this LID_ID is valid.
Definition lib_id.h:172
const UTF8 & GetLibItemName() const
Definition lib_id.h:102
const UTF8 & GetLibNickname() const
Return the logical library name portion of a LIB_ID.
Definition lib_id.h:87
virtual LIBRARY_MANAGER & GetLibraryManager() const
Definition pgm_base.h:134
wxString wx_str() const
Definition utf8.cpp:45
WX_PANEL(wxWindow *parent, wxWindowID id=wxID_ANY, const wxPoint &pos=wxDefaultPosition, const wxSize &size=wxSize(-1,-1), long style=wxTAB_TRAVERSAL, const wxString &name=wxEmptyString)
Definition wx_panel.cpp:28
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:150
bool IsOK(wxWindow *aParent, const wxString &aMessage)
Display a yes/no dialog with aMessage and returns the user response.
Definition confirm.cpp:259
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition confirm.cpp:202
void DisplayError(wxWindow *aParent, const wxString &aText)
Display an error or warning message box with aMessage.
Definition confirm.cpp:177
This file is part of the common library.
#define _(s)
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.
static const std::string KiCadDesignBlockLibPathExtension
static wxString KiCadDesignBlockLibPathWildcard()
std::unique_ptr< T > IO_RELEASER
Helper to hold and release an IO_BASE object when exceptions are thrown.
Definition io_mgr.h:33
LIBRARY_TABLE_SCOPE
PGM_BASE & Pgm()
The global program "get" accessor.
see class PGM_BASE
wxString message
Definition of file extensions used in Kicad.