KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
sheet.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) 2015 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
25#include <sch_draw_panel.h>
26#include <confirm.h>
27#include <kiface_base.h>
28#include <project.h>
29#include <math/vector2wx.h>
31#include <tool/tool_manager.h>
32#include <project_sch.h>
33#include <sch_edit_frame.h>
35#include <sch_sheet.h>
36#include <sch_sheet_path.h>
37#include <sch_view.h>
38#include <sch_painter.h>
39#include <schematic.h>
40#include <symbol_lib_table.h>
42#include <tool/actions.h>
43
44#include <wx/clipbrd.h>
45#include <wx/dcmemory.h>
46#include <wx/log.h>
47#include <wx/msgdlg.h>
48#include <wx/richmsgdlg.h>
49
50#include <advanced_config.h>
52
53
55{
56 wxASSERT( aSheet && aCurrentSheet );
57
58 wxString msg;
59 SCH_SHEET_LIST schematicSheets = Schematic().Hierarchy();
60 SCH_SHEET_LIST loadedSheets( aSheet ); // This is the schematicSheets of the loaded file.
61
62 wxString destFilePath = aCurrentSheet->LastScreen()->GetFileName();
63
64 if( destFilePath.IsEmpty() )
65 {
66 // If file is unsaved then there can't (yet) be any recursion.
67 return false;
68 }
69
70 // SCH_SCREEN object file paths are expected to be absolute. If this assert fires,
71 // something is seriously broken.
72 wxASSERT( wxFileName( destFilePath ).IsAbsolute() );
73
74 if( schematicSheets.TestForRecursion( loadedSheets, destFilePath ) )
75 {
76 msg.Printf( _( "The sheet changes cannot be made because the destination sheet already "
77 "has the sheet '%s' or one of its subsheets as a parent somewhere in the "
78 "schematic hierarchy." ),
79 destFilePath );
80 DisplayError( this, msg );
81 return true;
82 }
83
84 return false;
85}
86
87
89{
90 wxASSERT( aSheet && aSheet->GetScreen() );
91
92 wxString msg;
93 SCH_SCREENS newScreens( aSheet );
94
95 if( newScreens.HasNoFullyDefinedLibIds() )
96 {
97 msg.Printf( _( "The schematic '%s' has not had its symbol library links remapped "
98 "to the symbol library table. The project this schematic belongs to "
99 "must first be remapped before it can be imported into the current "
100 "project." ),
101 aSheet->GetScreen()->GetFileName() );
102 DisplayInfoMessage( this, msg );
103 return true;
104 }
105
106 return false;
107}
108
109
110void SCH_EDIT_FRAME::InitSheet( SCH_SHEET* aSheet, const wxString& aNewFilename )
111{
112 SCH_SCREEN* newScreen = new SCH_SCREEN( &Schematic() );
113 aSheet->SetScreen( newScreen );
114 aSheet->GetScreen()->SetContentModified();
115 aSheet->GetScreen()->SetFileName( aNewFilename );
116
118 wxCHECK( cfg, /* void */ );
119
121 newScreen->SetPageSettings( GetScreen()->GetPageSettings() );
122
123 const TITLE_BLOCK& tb1 = GetScreen()->GetTitleBlock();
124 TITLE_BLOCK tb2 = newScreen->GetTitleBlock();
125
127 tb2.SetRevision( tb1.GetRevision() );
128
129 if( cfg->m_PageSettings.export_date )
130 tb2.SetDate( tb1.GetDate() );
131
133 tb2.SetTitle( tb1.GetTitle() );
134
136 tb2.SetCompany( tb1.GetCompany() );
137
139 tb2.SetComment( 0, tb1.GetComment( 0 ) );
140
142 tb2.SetComment( 1, tb1.GetComment( 1 ) );
143
145 tb2.SetComment( 2, tb1.GetComment( 2 ) );
146
148 tb2.SetComment( 3, tb1.GetComment( 3 ) );
149
151 tb2.SetComment( 4, tb1.GetComment( 4 ) );
152
154 tb2.SetComment( 5, tb1.GetComment( 5 ) );
155
157 tb2.SetComment( 6, tb1.GetComment( 6 ) );
158
160 tb2.SetComment( 7, tb1.GetComment( 7 ) );
161
163 tb2.SetComment( 8, tb1.GetComment( 8 ) );
164
165 newScreen->SetTitleBlock( tb2 );
166}
167
168
170 const wxString& aFileName, bool aSkipRecursionCheck,
171 bool aSkipLibCheck )
172{
173 wxASSERT( aSheet && aCurrentSheet );
174
175 wxString msg;
176 wxFileName currentSheetFileName;
177 bool libTableChanged = false;
178
179 SCH_IO_MGR::SCH_FILE_T schFileType = SCH_IO_MGR::GuessPluginTypeFromSchPath( aFileName );
180
181 if( schFileType == SCH_IO_MGR::SCH_FILE_UNKNOWN )
182 schFileType = SCH_IO_MGR::SCH_KICAD;
183
184 IO_RELEASER<SCH_IO> pi( SCH_IO_MGR::FindPlugin( schFileType ) );
185 std::unique_ptr< SCH_SHEET> tmpSheet = std::make_unique<SCH_SHEET>( &Schematic() );
186
187 // This will cause the sheet UUID to be set to the UUID of the aSheet argument. This is
188 // required to ensure all of the sheet paths in any sub-sheets are correctly generated when
189 // using the temporary SCH_SHEET object that the file is loaded into..
190 const_cast<KIID&>( tmpSheet->m_Uuid ) = aSheet->m_Uuid;
191
192 wxFileName fileName( aFileName );
193
194 if( !fileName.IsAbsolute() && !fileName.MakeAbsolute() )
195 {
196 wxFAIL_MSG( wxString::Format( "Cannot make file name '%s' path absolute.", aFileName ) );
197 return false;
198 }
199
200 wxString fullFilename = fileName.GetFullPath();
201
202 try
203 {
204 if( aSheet->GetScreen() != nullptr )
205 {
206 tmpSheet.reset( pi->LoadSchematicFile( fullFilename, &Schematic() ) );
207 }
208 else
209 {
210 tmpSheet->SetFileName( fullFilename );
211 pi->LoadSchematicFile( fullFilename, &Schematic(), tmpSheet.get() );
212 }
213
214 if( !pi->GetError().IsEmpty() )
215 {
216 msg = _( "The entire schematic could not be loaded. Errors occurred attempting "
217 "to load hierarchical sheet schematics." );
218
219 wxMessageDialog msgDlg1( this, msg, _( "Schematic Load Error" ),
220 wxOK | wxCANCEL | wxCANCEL_DEFAULT |
221 wxCENTER | wxICON_QUESTION );
222 msgDlg1.SetOKLabel( wxMessageDialog::ButtonLabel( _( "Use partial schematic" ) ) );
223 msgDlg1.SetExtendedMessage( pi->GetError() );
224
225 if( msgDlg1.ShowModal() == wxID_CANCEL )
226 return false;
227 }
228 }
229 catch( const IO_ERROR& ioe )
230 {
231 msg.Printf( _( "Error loading schematic '%s'." ), fullFilename );
232 DisplayErrorMessage( this, msg, ioe.What() );
233
234 msg.Printf( _( "Failed to load '%s'." ), fullFilename );
235 SetMsgPanel( wxEmptyString, msg );
236
237 return false;
238 }
239
240 // If the loaded schematic is in a different folder from the current project and
241 // it contains hierarchical sheets, the hierarchical sheet paths need to be updated.
242 //
243 // Additionally, we need to make all backing screens absolute paths be in the current project
244 // path not the source path.
245 if( fileName.GetPathWithSep() != Prj().GetProjectPath() )
246 {
247 SCH_SHEET_LIST loadedSheets( tmpSheet.get() );
248
249 for( const SCH_SHEET_PATH& sheetPath : loadedSheets )
250 {
251 // Skip the loaded sheet since the user already determined if the file path should
252 // be relative or absolute.
253 if( sheetPath.size() == 1 )
254 continue;
255
256 wxString lastSheetPath = Prj().GetProjectPath();
257
258 for( unsigned i = 1; i < sheetPath.size(); i++ )
259 {
260 SCH_SHEET* sheet = sheetPath.at( i );
261 wxCHECK2( sheet, continue );
262
263 SCH_SCREEN* screen = sheet->GetScreen();
264 wxCHECK2( screen, continue );
265
266 // Use the screen file name which should always be absolute.
267 wxFileName loadedSheetFileName = screen->GetFileName();
268 wxCHECK2( loadedSheetFileName.IsAbsolute(), continue );
269
270 wxFileName tmp = loadedSheetFileName;
271 wxString sheetFileName;
272
273 if( tmp.MakeRelativeTo( lastSheetPath ) )
274 sheetFileName = tmp.GetFullPath();
275 else
276 sheetFileName = loadedSheetFileName.GetFullPath();
277
278 sheetFileName.Replace( wxT( "\\" ), wxT( "/" ) );
279 sheet->SetFileName( sheetFileName );
280 lastSheetPath = loadedSheetFileName.GetPath();
281 }
282 }
283 }
284
285 SCH_SHEET_LIST loadedSheets( tmpSheet.get() );
287 SCH_SHEET_LIST schematicSheets = Schematic().Hierarchy();
288
289 // Make sure any new sheet changes do not cause any recursion issues.
290 if( !aSkipRecursionCheck && CheckSheetForRecursion( tmpSheet.get(), aCurrentSheet ) )
291 return false;
292
293 if( checkForNoFullyDefinedLibIds( tmpSheet.get() ) )
294 return false;
295
296 // Make a valiant attempt to warn the user of all possible scenarios where there could
297 // be broken symbol library links.
298 wxArrayString names;
299 wxArrayString newLibNames;
300 SCH_SCREENS newScreens( tmpSheet.get() ); // All screens associated with the import.
301 SCH_SCREENS prjScreens( &Schematic().Root() );
302
303 newScreens.GetLibNicknames( names );
304
305 wxMessageDialog::ButtonLabel okButtonLabel( _( "Continue Load" ) );
306 wxMessageDialog::ButtonLabel cancelButtonLabel( _( "Cancel Load" ) );
307
308 // Prior to schematic file format 20221002, all symbol instance data was saved in the root
309 // sheet so loading a hierarchical sheet that is not the root sheet will have no symbol
310 // instance data. Give the user a chance to go back and save the project that contains this
311 // hierarchical sheet so the symbol instance data will be correct on load.
312 if( ( tmpSheet->GetScreen()->GetFileFormatVersionAtLoad() < 20221002 )
313 && tmpSheet->GetScreen()->GetSymbolInstances().empty() )
314 {
315 msg = _( "There are hierarchical sheets in the loaded schematic file from an older "
316 "file version resulting in missing symbol instance data. This will "
317 "result in all of the symbols in the loaded schematic to use either the "
318 "default instance setting or fall back to the library symbol settings. "
319 "Loading the project that uses this schematic file and saving to the "
320 "latest file version will resolve this issue.\n\n"
321 "Do you wish to continue?" );
322 wxMessageDialog msgDlg7( this, msg, _( "Continue Load Schematic" ),
323 wxOK | wxCANCEL | wxCANCEL_DEFAULT | wxCENTER | wxICON_QUESTION );
324 msgDlg7.SetOKCancelLabels( okButtonLabel, cancelButtonLabel );
325
326 if( msgDlg7.ShowModal() == wxID_CANCEL )
327 return false;
328 }
329
330 if( !aSkipLibCheck && !prjScreens.HasSchematic( fullFilename ) )
331 {
332 if( fileName.GetPathWithSep() == Prj().GetProjectPath() )
333 {
334 // A schematic in the current project path that isn't part of the current project.
335 // It's possible the user copied this schematic from another project so the library
336 // links may not be available. Even this is check is no guarantee that all symbol
337 // library links are valid but it's better than nothing.
338 for( const wxString& name : names )
339 {
340 if( !PROJECT_SCH::SchSymbolLibTable( &Prj() )->HasLibrary( name ) )
341 newLibNames.Add( name );
342 }
343
344 if( !newLibNames.IsEmpty() )
345 {
346 msg = _( "There are library names in the selected schematic that are missing "
347 "from the current project library table. This may result in broken "
348 "symbol library references for the loaded schematic.\n\n"
349 "Do you wish to continue?" );
350 wxMessageDialog msgDlg3( this, msg, _( "Continue Load Schematic" ),
351 wxOK | wxCANCEL | wxCANCEL_DEFAULT |
352 wxCENTER | wxICON_QUESTION );
353 msgDlg3.SetOKCancelLabels( okButtonLabel, cancelButtonLabel );
354
355 if( msgDlg3.ShowModal() == wxID_CANCEL )
356 return false;
357 }
358 }
359 else if( fileName.GetPathWithSep() != Prj().GetProjectPath() )
360 {
361 // A schematic loaded from a path other than the current project path.
362
363 // If there are symbol libraries in the imported schematic that are not in the
364 // symbol library table of this project, there could be a lot of broken symbol
365 // library links. Attempt to add the missing libraries to the project symbol
366 // library table.
367 wxArrayString duplicateLibNames;
368
369 for( const wxString& name : names )
370 {
371 if( !PROJECT_SCH::SchSymbolLibTable( &Prj() )->HasLibrary( name ) )
372 newLibNames.Add( name );
373 else
374 duplicateLibNames.Add( name );
375 }
376
378 wxFileName symLibTableFn( fileName.GetPath(),
380
381 // If there are any new or duplicate libraries, check to see if it's possible that
382 // there could be any missing libraries that would cause broken symbol library links.
383 if( !newLibNames.IsEmpty() || !duplicateLibNames.IsEmpty() )
384 {
385 if( !symLibTableFn.Exists() || !symLibTableFn.IsFileReadable() )
386 {
387 msg = _( "The selected file was created as part of a different project. "
388 "Linking the file to this project may result in missing or "
389 "incorrect symbol library references.\n\n"
390 "Do you wish to continue?" );
391 wxMessageDialog msgDlg4( this, msg, _( "Continue Load Schematic" ),
392 wxOK | wxCANCEL | wxCANCEL_DEFAULT | wxCENTER
393 | wxICON_QUESTION );
394 msgDlg4.SetOKCancelLabels( okButtonLabel, cancelButtonLabel );
395
396 if( msgDlg4.ShowModal() == wxID_CANCEL )
397 return false;
398 }
399 else
400 {
401 try
402 {
403 table.Load( symLibTableFn.GetFullPath() );
404 }
405 catch( const IO_ERROR& ioe )
406 {
407 msg.Printf( _( "Error loading the symbol library table '%s'." ),
408 symLibTableFn.GetFullPath() );
409 DisplayErrorMessage( nullptr, msg, ioe.What() );
410 return false;
411 }
412 }
413 }
414
415 // Check to see if any of the symbol libraries found in the appended schematic do
416 // not exist in the current project are missing from the appended project symbol
417 // library table.
418 if( !newLibNames.IsEmpty() )
419 {
420 bool missingLibNames = table.IsEmpty();
421
422 if( !missingLibNames )
423 {
424 for( const wxString& newLibName : newLibNames )
425 {
426 if( !table.HasLibrary( newLibName ) )
427 {
428 missingLibNames = true;
429 break;
430 }
431 }
432 }
433
434 if( missingLibNames )
435 {
436 msg = _( "There are symbol library names in the selected schematic that "
437 "are missing from the selected schematic project library table. "
438 "This may result in broken symbol library references.\n\n"
439 "Do you wish to continue?" );
440 wxMessageDialog msgDlg5( this, msg, _( "Continue Load Schematic" ),
441 wxOK | wxCANCEL | wxCANCEL_DEFAULT |
442 wxCENTER | wxICON_QUESTION );
443 msgDlg5.SetOKCancelLabels( okButtonLabel, cancelButtonLabel );
444
445 if( msgDlg5.ShowModal() == wxID_CANCEL )
446 return false;
447 }
448 }
449
450 // The library name already exists in the current project. Check to see if the
451 // duplicate name is the same library in the current project. If it's not, it's
452 // most likely that the symbol library links will be broken.
453 if( !duplicateLibNames.IsEmpty() && !table.IsEmpty() )
454 {
455 bool libNameConflict = false;
456
457 for( const wxString& duplicateLibName : duplicateLibNames )
458 {
459 const SYMBOL_LIB_TABLE_ROW* thisRow = nullptr;
460 const SYMBOL_LIB_TABLE_ROW* otherRow = nullptr;
461
462 if( PROJECT_SCH::SchSymbolLibTable( &Prj() )->HasLibrary( duplicateLibName ) )
463 thisRow = PROJECT_SCH::SchSymbolLibTable( &Prj() )->FindRow( duplicateLibName );
464
465 if( table.HasLibrary( duplicateLibName ) )
466 otherRow = table.FindRow( duplicateLibName );
467
468 // It's in the global library table so there is no conflict.
469 if( thisRow && !otherRow )
470 continue;
471
472 if( !thisRow || !otherRow )
473 continue;
474
475 wxFileName otherUriFileName;
476 wxString thisURI = thisRow->GetFullURI( true );
477 wxString otherURI = otherRow->GetFullURI( false);
478
479 if( otherURI.Contains( "${KIPRJMOD}" ) || otherURI.Contains( "$(KIPRJMOD)" ) )
480 {
481 // Cannot use relative paths here, "${KIPRJMOD}../path-to-cache-lib" does
482 // not expand to a valid symbol library path.
483 otherUriFileName.SetPath( fileName.GetPath() );
484 otherUriFileName.SetFullName( otherURI.AfterLast( '}' ) );
485 otherURI = otherUriFileName.GetFullPath();
486 }
487
488 if( thisURI != otherURI )
489 {
490 libNameConflict = true;
491 break;
492 }
493 }
494
495 if( libNameConflict )
496 {
497 msg = _( "A duplicate library name that references a different library exists "
498 "in the current library table. This conflict cannot be resolved and "
499 "may result in broken symbol library references.\n\n"
500 "Do you wish to continue?" );
501 wxMessageDialog msgDlg6( this, msg, _( "Continue Load Schematic" ),
502 wxOK | wxCANCEL | wxCANCEL_DEFAULT |
503 wxCENTER | wxICON_QUESTION );
504 msgDlg6.SetOKCancelLabels( okButtonLabel, cancelButtonLabel );
505
506 if( msgDlg6.ShowModal() == wxID_CANCEL )
507 return false;
508 }
509 }
510
511 // All (most?) of the possible broken symbol library link cases are covered. Map the
512 // new appended schematic project symbol library table entries to the current project
513 // symbol library table.
514 if( !newLibNames.IsEmpty() && !table.IsEmpty() )
515 {
516 for( const wxString& libName : newLibNames )
517 {
518 if( !table.HasLibrary( libName )
519 || PROJECT_SCH::SchSymbolLibTable( &Prj() )->HasLibrary( libName ) )
520 {
521 continue;
522 }
523
524 // Don't expand environment variable because KIPRJMOD will not be correct
525 // for a different project.
526 wxString uri = table.GetFullURI( libName, false );
527 wxFileName newLib;
528
529 if( uri.Contains( "${KIPRJMOD}" ) || uri.Contains( "$(KIPRJMOD)" ) )
530 {
531 // Cannot use relative paths here, "${KIPRJMOD}../path-to-cache-lib" does
532 // not expand to a valid symbol library path.
533 newLib.SetPath( fileName.GetPath() );
534 newLib.SetFullName( uri.AfterLast( '}' ) );
535 uri = newLib.GetFullPath();
536 }
537 else
538 {
539 uri = table.GetFullURI( libName );
540 }
541
542 // Add the library from the imported project to the current project
543 // symbol library table.
544 const SYMBOL_LIB_TABLE_ROW* row = table.FindRow( libName );
545
546 wxCHECK( row, false );
547
548 SYMBOL_LIB_TABLE_ROW* newRow = new SYMBOL_LIB_TABLE_ROW( libName, uri,
549 row->GetType(),
550 row->GetOptions(),
551 row->GetDescr() );
552
554 libTableChanged = true;
555 }
556 }
557 }
558 }
559
560 SCH_SCREEN* newScreen = tmpSheet->GetScreen();
561 wxCHECK_MSG( newScreen, false, "No screen defined for sheet." );
562
563 if( libTableChanged )
564 {
565 PROJECT_SCH::SchSymbolLibTable( &Prj() )->Save( Prj().GetProjectPath() +
567 }
568
569 // Make the best attempt to set the symbol instance data for the loaded schematic.
570 if( newScreen->GetFileFormatVersionAtLoad() < 20221002 )
571 {
572 if( !newScreen->GetSymbolInstances().empty() )
573 {
574 // If the loaded schematic is a root sheet for another project, update the symbol
575 // instances.
576 loadedSheets.UpdateSymbolInstanceData( newScreen->GetSymbolInstances());
577 }
578 }
579
580 newScreen->MigrateSimModels();
581
582 // Attempt to create new symbol instances using the instance data loaded above.
583 loadedSheets.AddNewSymbolInstances( *aCurrentSheet, Prj().GetProjectName() );
584
585 // Add new sheet instance data.
586 loadedSheets.AddNewSheetInstances( *aCurrentSheet, schematicSheets.GetLastVirtualPageNumber() );
587
588 // It is finally safe to add or append the imported schematic.
589 if( aSheet->GetScreen() == nullptr )
590 aSheet->SetScreen( newScreen );
591 else
592 aSheet->GetScreen()->Append( newScreen );
593
594 SCH_SCREENS allLoadedScreens( aSheet );
595 allLoadedScreens.ReplaceDuplicateTimeStamps();
596
597 return true;
598}
599
600
602 bool* aIsUndoable, bool* aClearAnnotationNewItems,
603 bool* aUpdateHierarchyNavigator,
604 wxString* aSourceSheetFilename )
605{
606 if( aSheet == nullptr || aHierarchy == nullptr )
607 return false;
608
609 // Get the new texts
610 DIALOG_SHEET_PROPERTIES dlg( this, aSheet, aIsUndoable, aClearAnnotationNewItems,
611 aUpdateHierarchyNavigator, aSourceSheetFilename );
612
613 if( dlg.ShowModal() == wxID_CANCEL )
614 return false;
615
617
618 return true;
619}
620
621
623{
624 wxRect drawArea;
625 BASE_SCREEN* screen = GetScreen();
626
627 drawArea.SetSize( ToWxSize( GetPageSizeIU() ) );
628
629 // Calculate a reasonable dc size, in pixels, and the dc scale to fit
630 // the drawings into the dc size
631 // scale is the ratio resolution (in PPI) / internal units
632 double ppi = 300; // Use 300 pixels per inch to create bitmap images on start
633 double inch2Iu = 1000.0 * schIUScale.IU_PER_MILS;
634 double scale = ppi / inch2Iu;
635
636 wxSize dcsize = drawArea.GetSize();
637
638 int maxdim = std::max( dcsize.x, dcsize.y );
639
640 // the max size in pixels of the bitmap used to build the sheet copy
641 const int maxbitmapsize = 5600;
642
643 while( int( maxdim * scale ) > maxbitmapsize )
644 {
645 ppi = ppi / 1.5;
646 scale = ppi / inch2Iu;
647 }
648
649 dcsize.x *= scale;
650 dcsize.y *= scale;
651
652 // Set draw offset, zoom... to values needed to draw in the memory DC
653 // after saving initial values:
654 VECTOR2I tmp_startvisu = screen->m_StartVisu;
655 VECTOR2I old_org = screen->m_DrawOrg;
656 screen->m_DrawOrg.x = screen->m_DrawOrg.y = 0;
657 screen->m_StartVisu.x = screen->m_StartVisu.y = 0;
658
659 wxMemoryDC dc;
660 wxBitmap image( dcsize );
661 dc.SelectObject( image );
662 dc.Clear();
663
664 GRResetPenAndBrush( &dc );
665 GRForceBlackPen( false );
666 dc.SetUserScale( scale, scale );
667
669
670 cfg->SetPrintDC( &dc );
671
672 // Init the color of the layer actually used to print the drawing sheet:
674
675 cfg->SetDefaultFont( eeconfig()->m_Appearance.default_font );
676
677 try
678 {
679 dc.SetUserScale( 1.0, 1.0 );
680 SCH_PRINTOUT printout( this, wxEmptyString );
681 // Ensure title block will be when printed on clipboard, regardless
682 // the current Cairo print option
683 EESCHEMA_SETTINGS* eecfg = eeconfig();
684 bool print_tb_opt = eecfg->m_Printing.title_block;
685 eecfg->m_Printing.title_block = true;
686 bool success = printout.PrintPage( GetScreen(), cfg->GetPrintDC(), false );
687 eecfg->m_Printing.title_block = print_tb_opt;
688
689 if( !success )
690 wxLogMessage( _( "Cannot create the schematic image") );
691 }
692 catch( ... )
693 {
694 wxLogMessage( "printout internal error" );
695 }
696
697 // Deselect Bitmap from DC before using the bitmap
698 dc.SelectObject( wxNullBitmap );
699
700 {
701 wxLogNull doNotLog; // disable logging of failed clipboard actions
702
703 if( wxTheClipboard->Open() )
704 {
705 // This data objects are held by the clipboard, so do not delete them in the app.
706 wxBitmapDataObject* clipbrd_data = new wxBitmapDataObject( image );
707 wxTheClipboard->SetData( clipbrd_data );
708 wxTheClipboard->Flush(); // Allow data to be available after closing KiCad
709 wxTheClipboard->Close();
710 }
711 }
712
713 GRForceBlackPen( false );
714
715 screen->m_StartVisu = tmp_startvisu;
716 screen->m_DrawOrg = old_org;
717}
718
719
721 const wxString& aSchematicFileName )
722{
723 wxString msg;
725 wxFileName fn = aSchematicFileName;
726
727 wxCHECK( fn.IsAbsolute(), false );
728
729 auto can_cause_issues = [&]() -> bool
730 {
731 wxFileName lhs;
732 wxFileName rhs = aSchematicFileName;
733 wxFileName old = aOldName;
734 wxString oldLower = old.GetFullName().Lower();
735 wxString rhsLower = rhs.GetFullName().Lower();
736 wxString lhsLower;
737
738 size_t count = 0;
739
740 wxCHECK( rhs.IsAbsolute(), false );
741
742 for( SCH_SHEET_PATH& sheet : sheets )
743 {
744 lhs = sheet.LastScreen()->GetFileName();
745
746 if( lhs.GetPath() != rhs.GetPath() )
747 continue;
748
749 lhsLower = lhs.GetFullName().Lower();
750
751 if( lhsLower == rhsLower && lhs.GetFullName() != rhs.GetFullName() )
752 count++;
753 }
754
755 // If we are renaming a sheet that is only used once, then we are not going to cause
756 // a case sensitivity issue.
757 if( oldLower == rhsLower )
758 return count > 1;
759
760 return count > 0;
761 };
762
763 if( eeconfig()->m_Appearance.show_sheet_filename_case_sensitivity_dialog && can_cause_issues() )
764 {
765 msg.Printf( _( "The file name '%s' can cause issues with an existing file name\n"
766 "already defined in the schematic on systems that support case\n"
767 "insensitive file names. This will cause issues if you copy this\n"
768 "project to an operating system that supports case insensitive file\n"
769 "names.\n\nDo you wish to continue?" ),
770 fn.GetName() );
771
772 wxRichMessageDialog dlg( this, msg, _( "Warning" ),
773 wxYES_NO | wxNO_DEFAULT | wxICON_QUESTION );
774 dlg.ShowCheckBox( _( "Do not show this message again." ) );
775 dlg.SetYesNoLabels( wxMessageDialog::ButtonLabel( _( "Create New Sheet" ) ),
776 wxMessageDialog::ButtonLabel( _( "Cancel" ) ) );
777
778 if( dlg.ShowModal() == wxID_NO )
779 return false;
780
782 !dlg.IsCheckBoxChecked();
783 }
784
785 return true;
786}
const char * name
Definition: DXF_plotter.cpp:59
constexpr EDA_IU_SCALE schIUScale
Definition: base_units.h:110
Handles how to draw a screen (a board, a schematic ...)
Definition: base_screen.h:41
VECTOR2I m_DrawOrg
offsets for drawing the circuit on the screen
Definition: base_screen.h:88
VECTOR2I m_StartVisu
Coordinates in drawing units of the current view position (upper left corner of device)
Definition: base_screen.h:93
void SetContentModified(bool aModified=true)
Definition: base_screen.h:59
int ShowModal() override
void SetMsgPanel(const std::vector< MSG_PANEL_ITEM > &aList)
Clear the message panel and populates it with the contents of aList.
const KIID m_Uuid
Definition: eda_item.h:498
PAGE_SETTINGS m_PageSettings
static const TOOL_EVENT SelectedItemsModified
Selected items were moved, this can be very high frequency on the canvas, use with care.
Definition: actions.h:343
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
void SetLayerColor(int aLayer, const COLOR4D &aColor)
Change the color used to draw a layer.
void SetDefaultFont(const wxString &aFont)
const COLOR4D & GetLayerColor(int aLayer) const
Return the color used to draw a layer.
void SetPrintDC(wxDC *aDC)
wxDC * GetPrintDC() const
Definition: kiid.h:49
const wxString & GetOptions() const
Return the options string, which may hold a password or anything else needed to instantiate the under...
const wxString & GetDescr() const
Return the description of the library referenced by this row.
const wxString GetFullURI(bool aSubstituted=false) const
Return the full location specifying URI for the LIB, either in original UI form or in environment var...
bool InsertRow(LIB_TABLE_ROW *aRow, bool doReplace=false)
Adds aRow if it does not already exist or if doReplace is true.
void Save(const wxString &aFileName) const
Write this library table to aFileName in s-expression form.
static SYMBOL_LIB_TABLE * SchSymbolLibTable(PROJECT *aProject)
Accessor for project symbol library table.
virtual const wxString GetProjectPath() const
Return the full path of the project.
Definition: project.cpp:146
SCH_SHEET_LIST Hierarchy() const
Return the full schematic flattened hierarchical sheet list.
Definition: schematic.cpp:208
void RefreshHierarchy()
Definition: schematic.cpp:216
SCH_RENDER_SETTINGS * GetRenderSettings()
const VECTOR2I GetPageSizeIU() const override
Works off of GetPageSettings() to return the size of the paper page in the internal units of this par...
EESCHEMA_SETTINGS * eeconfig() const
const PAGE_INFO & GetPageSettings() const override
bool AllowCaseSensitiveFileNameClashes(const wxString &aOldName, const wxString &aSchematicFileName)
Check aSchematicFileName for a potential file name case sensitivity clashes.
Definition: sheet.cpp:720
SCH_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
void InitSheet(SCH_SHEET *aSheet, const wxString &aNewFilename)
Definition: sheet.cpp:110
bool LoadSheetFromFile(SCH_SHEET *aSheet, SCH_SHEET_PATH *aCurrentSheet, const wxString &aFileName, bool aSkipRecursionCheck=false, bool aSkipLibCheck=false)
Load a the KiCad schematic file aFileName into the sheet aSheet.
Definition: sheet.cpp:169
SCHEMATIC & Schematic() const
void DrawCurrentSheetToClipboard()
Use the wxWidgets print code to draw an image of the current sheet onto the clipboard.
Definition: sheet.cpp:622
bool checkForNoFullyDefinedLibIds(SCH_SHEET *aSheet)
Verify that the symbol library links aSheet and all of its child sheets have been remapped to the sym...
Definition: sheet.cpp:88
bool EditSheetProperties(SCH_SHEET *aSheet, SCH_SHEET_PATH *aHierarchy, bool *aIsUndoable=nullptr, bool *aClearAnnotationNewItems=nullptr, bool *aUpdateHierarchyNavigator=nullptr, wxString *aSourceSheetFilename=nullptr)
Edit an existing sheet or add a new sheet to the schematic.
Definition: sheet.cpp:601
bool CheckSheetForRecursion(SCH_SHEET *aSheet, SCH_SHEET_PATH *aCurrentSheet)
Verify that aSheet will not cause a recursion error in aCurrentSheet.
Definition: sheet.cpp:54
static SCH_FILE_T GuessPluginTypeFromSchPath(const wxString &aSchematicPath, int aCtl=0)
Return a plugin type given a schematic using the file extension of aSchematicPath.
Definition: sch_io_mgr.cpp:179
Custom print out for printing schematics.
Definition: sch_printout.h:39
bool PrintPage(SCH_SCREEN *aScreen, wxDC *aDC, bool aForPrinting)
Print the current SCH_SCREEN using a given wxDC.
Container class that holds multiple SCH_SCREEN objects in a hierarchy.
Definition: sch_screen.h:734
bool HasNoFullyDefinedLibIds()
Test all of the schematic symbols to see if all LIB_ID objects library nickname is not set.
int ReplaceDuplicateTimeStamps()
Test all sheet and symbol objects in the schematic for duplicate time stamps and replaces them as nec...
bool HasSchematic(const wxString &aSchematicFileName)
Check if one of the schematics in the list of screens is aSchematicFileName.
size_t GetLibNicknames(wxArrayString &aLibNicknames)
Fetch all of the symbol library nicknames into aLibNicknames.
void SetTitleBlock(const TITLE_BLOCK &aTitleBlock)
Definition: sch_screen.h:160
void Append(SCH_ITEM *aItem, bool aUpdateLibSymbol=true)
Definition: sch_screen.cpp:155
void SetPageSettings(const PAGE_INFO &aPageSettings)
Definition: sch_screen.h:135
const wxString & GetFileName() const
Definition: sch_screen.h:147
void SetFileName(const wxString &aFileName)
Set the file name for this screen to aFileName.
Definition: sch_screen.cpp:120
const std::vector< SCH_SYMBOL_INSTANCE > & GetSymbolInstances() const
Definition: sch_screen.h:520
int GetFileFormatVersionAtLoad() const
Definition: sch_screen.h:132
const TITLE_BLOCK & GetTitleBlock() const
Definition: sch_screen.h:158
void MigrateSimModels()
Migrate any symbols having V6 simulation models to their V7 equivalents.
A container for handling SCH_SHEET_PATH objects in a flattened hierarchy.
void AddNewSymbolInstances(const SCH_SHEET_PATH &aPrefixSheetPath, const wxString &aProjectName)
Attempt to add new symbol instances for all symbols in this list of sheet paths prefixed with aPrefix...
int GetLastVirtualPageNumber() const
void UpdateSymbolInstanceData(const std::vector< SCH_SYMBOL_INSTANCE > &aSymbolInstances)
Update all of the symbol instance information using aSymbolInstances.
void AddNewSheetInstances(const SCH_SHEET_PATH &aPrefixSheetPath, int aLastVirtualPageNumber)
bool TestForRecursion(const SCH_SHEET_LIST &aSrcSheetHierarchy, const wxString &aDestFileName)
Test every SCH_SHEET_PATH in this SCH_SHEET_LIST to verify if adding the sheets stored in aSrcSheetHi...
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
SCH_SCREEN * LastScreen()
Sheet symbol placed in a schematic, and is the entry point for a sub schematic.
Definition: sch_sheet.h:47
void SetFileName(const wxString &aFilename)
Definition: sch_sheet.h:327
SCH_SCREEN * GetScreen() const
Definition: sch_sheet.h:116
void SetScreen(SCH_SCREEN *aScreen)
Set the SCH_SCREEN associated with this sheet to aScreen.
Definition: sch_sheet.cpp:137
Hold a record identifying a symbol library accessed by the appropriate symbol library SCH_IO object i...
const wxString GetType() const override
Return the type of symbol library table represented by this row.
static const wxString GetSymbolLibTableFileName()
SYMBOL_LIB_TABLE_ROW * FindRow(const wxString &aNickName, bool aCheckIfEnabled=false)
Return an SYMBOL_LIB_TABLE_ROW if aNickName is found in this table or in any chained fallBack table f...
Hold the information shown in the lower right corner of a plot, printout, or editing view.
Definition: title_block.h:41
const wxString & GetCompany() const
Definition: title_block.h:96
void SetRevision(const wxString &aRevision)
Definition: title_block.h:81
void SetComment(int aIdx, const wxString &aComment)
Definition: title_block.h:101
const wxString & GetRevision() const
Definition: title_block.h:86
void SetTitle(const wxString &aTitle)
Definition: title_block.h:58
const wxString & GetDate() const
Definition: title_block.h:76
const wxString & GetComment(int aIdx) const
Definition: title_block.h:107
void SetCompany(const wxString &aCompany)
Definition: title_block.h:91
const wxString & GetTitle() const
Definition: title_block.h:63
void SetDate(const wxString &aDate)
Set the date field, and defaults to the current time and date.
Definition: title_block.h:71
TOOL_MANAGER * m_toolManager
Definition: tools_holder.h:171
bool ProcessEvent(const TOOL_EVENT &aEvent)
Propagate an event to tools that requested events of matching type(s).
void DisplayInfoMessage(wxWindow *aParent, const wxString &aMessage, const wxString &aExtraInfo)
Display an informational message box with aMessage.
Definition: confirm.cpp:221
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)
void GRForceBlackPen(bool flagforce)
Definition: gr_basic.cpp:159
void GRResetPenAndBrush(wxDC *DC)
Definition: gr_basic.cpp:73
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:597
@ LAYER_DRAWINGSHEET
Sheet frame and title block.
Definition: layer_ids.h:277
@ LAYER_SCHEMATIC_DRAWINGSHEET
Definition: layer_ids.h:484
Definition of the SCH_SHEET_PATH and SCH_SHEET_LIST classes for Eeschema.
const int scale
bool title_block
Whether or not to print title block.
Definition: app_settings.h:160
const double IU_PER_MILS
Definition: base_units.h:77
wxSize ToWxSize(const VECTOR2I &aSize)
Definition: vector2wx.h:55
Definition of file extensions used in Kicad.