KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
sch_design_block_utils.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24#include <pgm_base.h>
25#include <kiway.h>
26#include <design_block.h>
29#include <sch_edit_frame.h>
30#include <wx/choicdlg.h>
31#include <wx/msgdlg.h>
32#include <wx/textdlg.h>
34#include <paths.h>
35#include <env_paths.h>
36#include <common.h>
37#include <kidialog.h>
38#include <confirm.h>
39#include <tool/tool_manager.h>
40#include <sch_selection_tool.h>
42#include <nlohmann/json.hpp>
43
44bool checkOverwriteDb( wxWindow* aFrame, wxString& libname, wxString& newName )
45{
46 wxString msg = wxString::Format( _( "Design block '%s' already exists in library '%s'." ), newName.GetData(),
47 libname.GetData() );
48
49 if( OKOrCancelDialog( aFrame, _( "Confirmation" ), msg, _( "Overwrite existing design block?" ), _( "Overwrite" ) )
50 != wxID_OK )
51 {
52 return false;
53 }
54
55 return true;
56}
57
58
59bool checkOverwriteDbSchematic( wxWindow* aFrame, const LIB_ID& aLibId )
60{
61 wxString msg =
62 wxString::Format( _( "Design block '%s' already has a schematic." ), aLibId.GetUniStringLibItemName() );
63
64 if( OKOrCancelDialog( aFrame, _( "Confirmation" ), msg, _( "Overwrite existing schematic?" ), _( "Overwrite" ) )
65 != wxID_OK )
66 {
67 return false;
68 }
69
70 return true;
71}
72
73
74bool SCH_EDIT_FRAME::SaveSheetAsDesignBlock( const wxString& aLibraryName, SCH_SHEET_PATH& aSheetPath )
75{
76 // Make sure the user has selected a library to save into
78 {
79 DisplayErrorMessage( this, _( "Please select a library to save the design block to." ) );
80 return false;
81 }
82
83 // Just block all attempts to create design blocks with nested sheets at this point
84 std::vector<SCH_ITEM*> sheets;
85 aSheetPath.LastScreen()->GetSheets( &sheets );
86
87 if( !sheets.empty() )
88 {
89 DisplayErrorMessage( this, _( "Design blocks with nested sheets are not supported." ) );
90 return false;
91 }
92
93 DESIGN_BLOCK blk;
94 wxFileName fn = wxFileNameFromPath( aSheetPath.Last()->GetName() );
95
96 blk.SetLibId( LIB_ID( aLibraryName, fn.GetName() ) );
97
98 // Copy all fields from the sheet to the design block
99 std::vector<SCH_FIELD>& shFields = aSheetPath.Last()->GetFields();
100 nlohmann::ordered_map<wxString, wxString> dbFields;
101
102 for( SCH_FIELD& f : shFields )
103 {
104 if( f.GetId() == FIELD_T::SHEET_NAME || f.GetId() == FIELD_T::SHEET_FILENAME )
105 continue;
106
107 dbFields[f.GetCanonicalName()] = f.GetText();
108 }
109
110 blk.SetFields( dbFields );
111
112 DIALOG_DESIGN_BLOCK_PROPERTIES dlg( this, &blk );
113
114 if( dlg.ShowModal() != wxID_OK )
115 return false;
116
117 wxString libName = blk.GetLibId().GetLibNickname();
118 wxString newName = blk.GetLibId().GetLibItemName();
119
120 if( Prj().DesignBlockLibs()->DesignBlockExists( libName, newName ) && !checkOverwriteDb( this, libName, newName ) )
121 {
122 return false;
123 }
124
125 // Save a temporary copy of the schematic file, as the plugin is just going to move it
126 wxString tempFile = wxFileName::CreateTempFileName( "design_block" );
127 if( !saveSchematicFile( aSheetPath.Last(), tempFile ) )
128 {
129 DisplayErrorMessage( this, _( "Error saving temporary schematic file to create design block." ) );
130 wxRemoveFile( tempFile );
131 return false;
132 }
133
134 blk.SetSchematicFile( tempFile );
135
136 bool success = false;
137
138 try
139 {
140 success = Prj().DesignBlockLibs()->DesignBlockSave( aLibraryName, &blk ) == DESIGN_BLOCK_LIB_TABLE::SAVE_OK;
141 }
142 catch( const IO_ERROR& ioe )
143 {
144 DisplayError( this, ioe.What() );
145 }
146
147 // Clean up the temporary file
148 wxRemoveFile( tempFile );
149
152
153 return success;
154}
155
156
158{
159 // Make sure the user has selected a library to save into
160 if( !Prj().DesignBlockLibs()->DesignBlockExists( aLibId.GetLibNickname(), aLibId.GetLibItemName() ) )
161 {
162 DisplayErrorMessage( this, _( "Please select a design block to save the schematic to." ) );
163 return false;
164 }
165
166 // Just block all attempts to create design blocks with nested sheets at this point
167 std::vector<SCH_ITEM*> sheets;
168 aSheetPath.LastScreen()->GetSheets( &sheets );
169
170 if( !sheets.empty() )
171 {
172 DisplayErrorMessage( this, _( "Design blocks with nested sheets are not supported." ) );
173 return false;
174 }
175
176 DESIGN_BLOCK* blk = nullptr;
177
178 try
179 {
180 blk = Prj().DesignBlockLibs()->DesignBlockLoad( aLibId.GetLibNickname(), aLibId.GetLibItemName() );
181 }
182 catch( const IO_ERROR& ioe )
183 {
184 DisplayError( this, ioe.What() );
185 return false;
186 }
187
188 if( !blk->GetSchematicFile().IsEmpty() && !checkOverwriteDbSchematic( this, aLibId ) )
189 {
190 return false;
191 }
192
193 // Copy all fields from the sheet to the design block.
194 // Note: this will overwrite any existing fields in the design block, but
195 // will leave extra fields not in this source sheet alone.
196 std::vector<SCH_FIELD>& shFields = aSheetPath.Last()->GetFields();
197 nlohmann::ordered_map<wxString, wxString> dbFields = blk->GetFields();
198
199 for( SCH_FIELD& f : shFields )
200 {
201 if( f.GetId() == FIELD_T::SHEET_NAME || f.GetId() == FIELD_T::SHEET_FILENAME )
202 continue;
203
204 dbFields[f.GetCanonicalName()] = f.GetText();
205 }
206
207 blk->SetFields( dbFields );
208
209 DIALOG_DESIGN_BLOCK_PROPERTIES dlg( this, blk, true );
210
211 if( dlg.ShowModal() != wxID_OK )
212 return false;
213
214 // Save a temporary copy of the schematic file, as the plugin is just going to move it
215 wxString tempFile = wxFileName::CreateTempFileName( "design_block" );
216 if( !saveSchematicFile( aSheetPath.Last(), tempFile ) )
217 {
218 DisplayErrorMessage( this, _( "Error saving temporary schematic file to create design block." ) );
219 wxRemoveFile( tempFile );
220 return false;
221 }
222
223 blk->SetSchematicFile( tempFile );
224
225 bool success = false;
226
227 try
228 {
229 success = Prj().DesignBlockLibs()->DesignBlockSave( aLibId.GetLibNickname(), blk )
231 }
232 catch( const IO_ERROR& ioe )
233 {
234 DisplayError( this, ioe.What() );
235 }
236
237 // Clean up the temporary file
238 wxRemoveFile( tempFile );
239
242
243 return success;
244}
245
246
247bool SCH_EDIT_FRAME::SaveSelectionAsDesignBlock( const wxString& aLibraryName )
248{
249 // Get all selected items
250 SCH_SELECTION selection = m_toolManager->GetTool<SCH_SELECTION_TOOL>()->GetSelection();
251
252 if( selection.Empty() )
253 {
254 DisplayErrorMessage( this, _( "Please select some items to save as a design block." ) );
255 return false;
256 }
257
258 // Make sure the user has selected a library to save into
260 {
261 DisplayErrorMessage( this, _( "Please select a library to save the design block to." ) );
262 return false;
263 }
264
265 // Just block all attempts to create design blocks with nested sheets at this point
266 if( selection.HasType( SCH_SHEET_T ) )
267 {
268 if( selection.Size() == 1 )
269 {
270 SCH_SHEET* sheet = static_cast<SCH_SHEET*>( selection.Front() );
272
273 curPath.push_back( sheet );
274 SaveSheetAsDesignBlock( aLibraryName, curPath );
275 }
276 else
277 DisplayErrorMessage( this, _( "Design blocks with nested sheets are not supported." ) );
278
279 return false;
280 }
281
282 DESIGN_BLOCK blk;
283 wxFileName fn = wxFileNameFromPath( GetScreen()->GetFileName() );
284
285 blk.SetLibId( LIB_ID( aLibraryName, fn.GetName() ) );
286
287 DIALOG_DESIGN_BLOCK_PROPERTIES dlg( this, &blk );
288
289 if( dlg.ShowModal() != wxID_OK )
290 return false;
291
292 wxString libName = blk.GetLibId().GetLibNickname();
293 wxString newName = blk.GetLibId().GetLibItemName();
294
295 if( Prj().DesignBlockLibs()->DesignBlockExists( libName, newName ) && !checkOverwriteDb( this, libName, newName ) )
296 {
297 return false;
298 }
299
300 // Create a temporary screen
301 SCH_SCREEN* tempScreen = new SCH_SCREEN( m_schematic );
302
303 // Copy the selected items to the temporary screen
304 for( EDA_ITEM* item : selection )
305 {
306 EDA_ITEM* copy = item->Clone();
307 tempScreen->Append( static_cast<SCH_ITEM*>( copy ) );
308 }
309
310 // Create a sheet for the temporary screen
311 SCH_SHEET* tempSheet = new SCH_SHEET( m_schematic );
312 tempSheet->SetScreen( tempScreen );
313
314 // Save a temporary copy of the schematic file, as the plugin is just going to move it
315 wxString tempFile = wxFileName::CreateTempFileName( "design_block" );
316 if( !saveSchematicFile( tempSheet, tempFile ) )
317 {
318 DisplayErrorMessage( this, _( "Error saving temporary schematic file to create design block." ) );
319 wxRemoveFile( tempFile );
320 return false;
321 }
322
323 blk.SetSchematicFile( tempFile );
324
325 bool success = false;
326
327 try
328 {
329 success = Prj().DesignBlockLibs()->DesignBlockSave( aLibraryName, &blk ) == DESIGN_BLOCK_LIB_TABLE::SAVE_OK;
330 }
331 catch( const IO_ERROR& ioe )
332 {
333 DisplayError( this, ioe.What() );
334 }
335
336 // Clean up the temporaries
337 wxRemoveFile( tempFile );
338 // This will also delete the screen
339 delete tempSheet;
340
343
344 return success;
345}
346
347
349{
350 // Get all selected items
351 SCH_SELECTION selection = m_toolManager->GetTool<SCH_SELECTION_TOOL>()->GetSelection();
352
353 if( selection.Empty() )
354 {
355 DisplayErrorMessage( this, _( "Please select some items to save as a design block." ) );
356 return false;
357 }
358
359 // Make sure the user has selected a library to save into
360 if( !Prj().DesignBlockLibs()->DesignBlockExists( aLibId.GetLibNickname(), aLibId.GetLibItemName() ) )
361 {
362 DisplayErrorMessage( this, _( "Please select a design block to save the schematic to." ) );
363 return false;
364 }
365
366 // Just block all attempts to create design blocks with nested sheets at this point
367 if( selection.HasType( SCH_SHEET_T ) )
368 {
369 if( selection.Size() == 1 )
370 {
371 SCH_SHEET* sheet = static_cast<SCH_SHEET*>( selection.Front() );
373
374 curPath.push_back( sheet );
375 SaveSheetToDesignBlock( aLibId, curPath );
376 }
377 else
378 DisplayErrorMessage( this, _( "Design blocks with nested sheets are not supported." ) );
379
380 return false;
381 }
382
383 DESIGN_BLOCK* blk = nullptr;
384
385 try
386 {
387 blk = Prj().DesignBlockLibs()->DesignBlockLoad( aLibId.GetLibNickname(), aLibId.GetLibItemName() );
388 }
389 catch( const IO_ERROR& ioe )
390 {
391 DisplayError( this, ioe.What() );
392 return false;
393 }
394
395 if( !blk->GetSchematicFile().IsEmpty() && !checkOverwriteDbSchematic( this, aLibId ) )
396 {
397 return false;
398 }
399
400 // Create a temporary screen
401 SCH_SCREEN* tempScreen = new SCH_SCREEN( m_schematic );
402
403 // Copy the selected items to the temporary screen
404 for( EDA_ITEM* item : selection )
405 {
406 EDA_ITEM* copy = item->Clone();
407 tempScreen->Append( static_cast<SCH_ITEM*>( copy ) );
408 }
409
410 // Create a sheet for the temporary screen
411 SCH_SHEET* tempSheet = new SCH_SHEET( m_schematic );
412 tempSheet->SetScreen( tempScreen );
413
414 // Save a temporary copy of the schematic file, as the plugin is just going to move it
415 wxString tempFile = wxFileName::CreateTempFileName( "design_block" );
416 if( !saveSchematicFile( tempSheet, tempFile ) )
417 {
418 DisplayErrorMessage( this, _( "Error saving temporary schematic file to create design block." ) );
419 wxRemoveFile( tempFile );
420 return false;
421 }
422
423 blk->SetSchematicFile( tempFile );
424
425 bool success = false;
426
427 try
428 {
429 success = Prj().DesignBlockLibs()->DesignBlockSave( aLibId.GetLibNickname(), blk )
431 }
432 catch( const IO_ERROR& ioe )
433 {
434 DisplayError( this, ioe.What() );
435 }
436
437 // Clean up the temporaries
438 wxRemoveFile( tempFile );
439 // This will also delete the screen
440 delete tempSheet;
441
444
445 return success;
446}
DESIGN_BLOCK * DesignBlockLoad(const wxString &aNickname, const wxString &aDesignBlockName, bool aKeepUUID=false)
Load a design block having aDesignBlockName from the library given by aNickname.
SAVE_T DesignBlockSave(const wxString &aNickname, const DESIGN_BLOCK *aDesignBlock, bool aOverwrite=true)
Write aDesignBlock to an existing library given by aNickname.
void SelectLibId(const LIB_ID &aLibId)
LIB_ID GetSelectedLibId(int *aUnit=nullptr) const
void SetSchematicFile(const wxString &aFile)
Definition: design_block.h:46
void SetLibId(const LIB_ID &aName)
Definition: design_block.h:36
const wxString & GetSchematicFile() const
Definition: design_block.h:45
void SetFields(nlohmann::ordered_map< wxString, wxString > &aFields)
Definition: design_block.h:51
const LIB_ID & GetLibId() const
Definition: design_block.h:37
const nlohmann::ordered_map< wxString, wxString > & GetFields() const
Definition: design_block.h:56
int ShowModal() override
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
Definition: ki_exception.h:77
virtual const wxString What() const
A composite of Problem() and Where()
Definition: exceptions.cpp:30
A logical library item identifier and consists of various portions much like a URI.
Definition: lib_id.h:49
const wxString GetUniStringLibItemName() const
Get strings for display messages in dialogs.
Definition: lib_id.h:112
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 DESIGN_BLOCK_LIB_TABLE * DesignBlockLibs()
Return the table of design block libraries.
Definition: project.cpp:429
bool SaveSheetToDesignBlock(const LIB_ID &aLibId, SCH_SHEET_PATH &aSheetPath)
SCH_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
bool SaveSelectionToDesignBlock(const LIB_ID &aLibId)
SCHEMATIC * m_schematic
The currently loaded schematic.
SCH_SHEET_PATH & GetCurrentSheet() const
bool saveSchematicFile(SCH_SHEET *aSheet, const wxString &aSavePath)
Save aSheet to a schematic file.
SCH_DESIGN_BLOCK_PANE * m_designBlocksPane
bool SaveSheetAsDesignBlock(const wxString &aLibraryName, SCH_SHEET_PATH &aSheetPath)
bool SaveSelectionAsDesignBlock(const wxString &aLibraryName)
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
void Append(SCH_ITEM *aItem, bool aUpdateLibSymbol=true)
Definition: sch_screen.cpp:155
void GetSheets(std::vector< SCH_ITEM * > *aItems) const
Similar to Items().OfType( SCH_SHEET_T ), but return the sheets in a deterministic order (L-R,...
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
SCH_SCREEN * LastScreen()
SCH_SHEET * Last() const
Return a pointer to the last SCH_SHEET of the list.
void push_back(SCH_SHEET *aSheet)
Forwarded method from std::vector.
Sheet symbol placed in a schematic, and is the entry point for a sub schematic.
Definition: sch_sheet.h:47
std::vector< SCH_FIELD > & GetFields()
Return a reference to the vector holding the sheet's fields.
Definition: sch_sheet.h:87
wxString GetName() const
Definition: sch_sheet.h:113
void SetScreen(SCH_SCREEN *aScreen)
Set the SCH_SCREEN associated with this sheet to aScreen.
Definition: sch_sheet.cpp:137
EDA_ITEM * Front() const
Definition: selection.h:172
bool HasType(KICAD_T aType) const
Checks if there is at least one item of requested kind.
Definition: selection.cpp:143
int Size() const
Returns the number of selected parts.
Definition: selection.h:116
bool Empty() const
Checks if there is anything selected.
Definition: selection.h:110
TOOL_MANAGER * m_toolManager
Definition: tools_holder.h:171
bool empty() const
Definition: utf8.h:104
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:143
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition: confirm.cpp:194
void DisplayError(wxWindow *aParent, const wxString &aText)
Display an error or warning message box with aMessage.
Definition: confirm.cpp:170
This file is part of the common library.
#define _(s)
Helper functions to substitute paths with environmental variables.
PROJECT & Prj()
Definition: kicad.cpp:597
This file is part of the common library.
see class PGM_BASE
bool checkOverwriteDb(wxWindow *aFrame, wxString &libname, wxString &newName)
bool checkOverwriteDbSchematic(wxWindow *aFrame, const LIB_ID &aLibId)
@ SCH_SHEET_T
Definition: typeinfo.h:175
Definition of file extensions used in Kicad.