KiCad PCB EDA Suite
Loading...
Searching...
No Matches
gerbview/files.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) 2017 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, see <https://www.gnu.org/licenses/>.
19 */
20
21#include <wx/debug.h>
22#include <wx/filedlg.h>
23#include <wx/wfstream.h>
24#include <wx/zipstrm.h>
25#include <reporter.h>
27#include <gerbview_frame.h>
28#include <gerbview_id.h>
29#include <gerber_file_image.h>
31#include <excellon_image.h>
32#include <lset.h>
34#include <view/view.h>
37#include <tool/tool_manager.h>
38#include <kiplatform/ui.h>
39
40// HTML Messages used more than one time:
41#define MSG_NO_MORE_LAYER _( "<b>No more available layers</b> in GerbView to load files" )
42#define MSG_NOT_LOADED _( "<b>Not loaded:</b> <i>%s</i>" )
43#define MSG_OOM _( "<b>Memory was exhausted reading:</b> <i>%s</i>" )
44
45
46void GERBVIEW_FRAME::OnGbrFileHistory( wxCommandEvent& event )
47{
48 wxString filename = GetFileFromHistory( event.GetId(), _( "Gerber files" ) );
49
50 if( !filename.IsEmpty() )
51 LoadGerberFiles( filename );
52}
53
54void GERBVIEW_FRAME::OnClearGbrFileHistory( wxCommandEvent& aEvent )
55{
57}
58
59
60void GERBVIEW_FRAME::OnDrlFileHistory( wxCommandEvent& event )
61{
62 wxString filename = GetFileFromHistory( event.GetId(), _( "Drill files" ), &m_drillFileHistory );
63
64 if( !filename.IsEmpty() )
65 LoadExcellonFiles( filename );
66}
67
68
69void GERBVIEW_FRAME::OnClearDrlFileHistory( wxCommandEvent& aEvent )
70{
71 m_drillFileHistory.ClearFileHistory();
72
73 if( GetMenuBar() )
74 {
76 GetMenuBar()->Refresh();
77 }
78}
79
80
81void GERBVIEW_FRAME::OnZipFileHistory( wxCommandEvent& event )
82{
83 wxString filename = GetFileFromHistory( event.GetId(), _( "Zip files" ), &m_zipFileHistory );
84
85 if( !filename.IsEmpty() )
86 LoadZipArchiveFile( filename );
87}
88
89
90void GERBVIEW_FRAME::OnClearZipFileHistory( wxCommandEvent& aEvent )
91{
92 m_zipFileHistory.ClearFileHistory();
93
94 if( GetMenuBar() )
95 {
97 GetMenuBar()->Refresh();
98 }
99}
100
101
102void GERBVIEW_FRAME::OnJobFileHistory( wxCommandEvent& event )
103{
104 wxString filename = GetFileFromHistory( event.GetId(), _( "Job files" ), &m_jobFileHistory );
105
106 if( !filename.IsEmpty() )
107 LoadGerberJobFile( filename );
108}
109
110
111void GERBVIEW_FRAME::OnClearJobFileHistory( wxCommandEvent& aEvent )
112{
113 m_jobFileHistory.ClearFileHistory();
114
115 if( GetMenuBar() )
116 {
118 GetMenuBar()->Refresh();
119 }
120}
121
122
123bool GERBVIEW_FRAME::LoadFileOrShowDialog( const wxString& aFileName,
124 const wxString& dialogFiletypes,
125 const wxString& dialogTitle, const int filetype )
126{
127 static int lastGerberFileWildcard = 0;
128 wxArrayString filenamesList;
129 wxFileName filename = aFileName;
130 wxString currentPath;
131
132 if( !filename.IsOk() )
133 {
134 // Use the current working directory if the file name path does not exist.
135 if( filename.DirExists() )
136 currentPath = filename.GetPath();
137 else
138 {
139 currentPath = m_mruPath;
140
141 // On wxWidgets 3.1 (bug?) the path in wxFileDialog is ignored when
142 // finishing by the dir separator. Remove it if any:
143 if( currentPath.EndsWith( '\\' ) || currentPath.EndsWith( '/' ) )
144 currentPath.RemoveLast();
145 }
146
147 wxFileDialog dlg( this, dialogTitle, currentPath, filename.GetFullName(), dialogFiletypes,
148 wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE | wxFD_CHANGE_DIR );
149
151
152 wxArrayString dummy1, dummy2;
153 const int nWildcards = wxParseCommonDialogsFilter( dialogFiletypes, dummy1, dummy2 );
154
155 if( lastGerberFileWildcard >= 0 && lastGerberFileWildcard < nWildcards )
156 dlg.SetFilterIndex( lastGerberFileWildcard );
157
158 if( dlg.ShowModal() == wxID_CANCEL )
159 return false;
160
161 lastGerberFileWildcard = dlg.GetFilterIndex();
162 dlg.GetPaths( filenamesList );
163 m_mruPath = currentPath = dlg.GetDirectory();
164 }
165 else
166 {
167 filenamesList.Add( aFileName );
168 currentPath = filename.GetPath();
169 m_mruPath = currentPath;
170 }
171
172 // Set the busy cursor
173 wxBusyCursor wait;
174
175 bool isFirstFile = GetImagesList()->GetLoadedImageCount() == 0;
176
177 std::vector<int> fileTypesVec( filenamesList.Count(), filetype );
178 bool success = LoadListOfGerberAndDrillFiles( currentPath, filenamesList, &fileTypesVec );
179
180 // Auto zoom / sort is only applied when no other files have been loaded
181 if( isFirstFile )
182 {
183 int ly = GetActiveLayer();
184
186 Zoom_Automatique( false );
187
188 // Ensure the initial active graphic layer is updated after sorting.
189 SetActiveLayer( ly, true );
190 }
191
192 return success;
193}
194
195
196bool GERBVIEW_FRAME::LoadAutodetectedFiles( const wxString& aFileName )
197{
198 // 2 = autodetect files
199 return LoadFileOrShowDialog( aFileName, FILEEXT::AllFilesWildcard(), _( "Open Autodetected File(s)" ),
200 2 );
201}
202
203
204bool GERBVIEW_FRAME::LoadGerberFiles( const wxString& aFileName )
205{
206 wxString filetypes;
207 wxFileName filename = aFileName;
208
209 /* Standard gerber filetypes
210 * (See http://en.wikipedia.org/wiki/Gerber_File)
211 * The .gbr (.pho in legacy files) extension is the default used in Pcbnew; however
212 * there are a lot of other extensions used for gerber files. Because the first letter
213 * is usually g, we accept g* as extension.
214 * (Mainly internal copper layers do not have specific extension, and filenames are like
215 * *.g1, *.g2 *.gb1 ...)
216 * Now (2014) Ucamco (the company which manages the Gerber format) encourages use of .gbr
217 * only and the Gerber X2 file format.
218 */
219 filetypes = _( "Gerber files" ) + AddFileExtListToFilter( { "g*", "pho" } ) + wxT( "|" );
220
221 /* Special gerber filetypes */
222 filetypes += _( "Top layer" ) + AddFileExtListToFilter( { "gtl" } ) + wxT( "|" );
223 filetypes += _( "Bottom layer" ) + AddFileExtListToFilter( { "gbl" } ) + wxT( "|" );
224 filetypes += _( "Bottom solder resist" ) + AddFileExtListToFilter( { "gbs" } ) + wxT( "|" );
225 filetypes += _( "Top solder resist" ) + AddFileExtListToFilter( { "gts" } ) + wxT( "|" );
226 filetypes += _( "Bottom overlay" ) + AddFileExtListToFilter( { "gbo" } ) + wxT( "|" );
227 filetypes += _( "Top overlay" ) + AddFileExtListToFilter( { "gto" } ) + wxT( "|" );
228 filetypes += _( "Bottom paste" ) + AddFileExtListToFilter( { "gbp" } ) + wxT( "|" );
229 filetypes += _( "Top paste" ) + AddFileExtListToFilter( { "gtp" } ) + wxT( "|" );
230 filetypes += _( "Keep-out layer" ) + AddFileExtListToFilter( { "gko" } ) + wxT( "|" );
231 filetypes += _( "Mechanical layers" )
233 { "gm1", "gm2", "gm3", "gm4", "gm5", "gm6", "gm7", "gm8", "gm9" } )
234 + wxT( "|" );
235 filetypes += _( "Top Pad Master" ) + AddFileExtListToFilter( { "gpt" } ) + wxT( "|" );
236 filetypes += _( "Bottom Pad Master" ) + AddFileExtListToFilter( { "gpb" } ) + wxT( "|" );
237
238 // All filetypes
239 filetypes += FILEEXT::AllFilesWildcard();
240
241 // 0 = gerber files
242 return LoadFileOrShowDialog( aFileName, filetypes, _( "Open Gerber File(s)" ), 0 );
243}
244
245
246bool GERBVIEW_FRAME::LoadExcellonFiles( const wxString& aFileName )
247{
248 wxString filetypes = FILEEXT::DrillFileWildcard();
249 filetypes << wxT( "|" );
250 filetypes += FILEEXT::AllFilesWildcard();
251
252 // 1 = drill files
253 return LoadFileOrShowDialog( aFileName, filetypes, _( "Open NC (Excellon) Drill File(s)" ), 1 );
254}
255
256
258 const wxArrayString& aFilenameList,
259 std::vector<int>* aFileType )
260{
261 wxCHECK_MSG( aFilenameList.Count() == aFileType->size(), false,
262 "Mismatch in file names and file types count" );
263
264 wxFileName filename;
265
266 // Read gerber files: each file is loaded on a new GerbView layer
267 bool success = true;
268 int layer = GetActiveLayer();
269 int firstLoadedLayer = NO_AVAILABLE_LAYERS;
271
272 // Manage errors when loading files
274
275 // Create progress dialog (only used if more than 1 file to load
276 std::unique_ptr<WX_PROGRESS_REPORTER> progress = nullptr;
277
278 for( unsigned ii = 0; ii < aFilenameList.GetCount(); ii++ )
279 {
280 filename = aFilenameList[ii];
281
282 if( !filename.IsAbsolute() )
283 filename.SetPath( aPath );
284
285 // Check for non existing files, to avoid creating broken or useless data
286 // and report all in one error list:
287 if( !filename.FileExists() )
288 {
289 wxString warning;
290 warning << wxT( "<b>" ) << _( "File not found:" ) << wxT( "</b><br>" )
291 << filename.GetFullPath() << wxT( "<br>" );
292 reporter.Report( warning, RPT_SEVERITY_WARNING );
293 success = false;
294 continue;
295 }
296
297 if( filename.GetExt() == FILEEXT::GerberJobFileExtension.c_str() )
298 {
299 //We cannot read a gerber job file as a gerber plot file: skip it
300 wxString txt;
301 txt.Printf( _( "<b>A gerber job file cannot be loaded as a plot file</b> "
302 "<i>%s</i>" ),
303 filename.GetFullName() );
304 success = false;
305 reporter.Report( txt, RPT_SEVERITY_ERROR );
306 continue;
307 }
308
309
310 m_lastFileName = filename.GetFullPath();
311
312 if( !progress && ( aFilenameList.GetCount() > 1 ) )
313 {
314 progress = std::make_unique<WX_PROGRESS_REPORTER>( this, _( "Load Files" ), 1, PR_CAN_ABORT );
315 progress->SetMaxProgress( aFilenameList.GetCount() - 1 );
316 progress->Report( wxString::Format( _("Loading %u/%zu %s..." ),
317 ii+1,
318 aFilenameList.GetCount(),
319 m_lastFileName ) );
320 }
321 else if( progress )
322 {
323 progress->Report( wxString::Format( _("Loading %u/%zu %s..." ),
324 ii+1,
325 aFilenameList.GetCount(),
326 m_lastFileName ) );
327 progress->KeepRefreshing();
328 }
329
330
331 // Make sure we have a layer available to load into
332 layer = getNextAvailableLayer();
333
334 if( layer == NO_AVAILABLE_LAYERS )
335 {
336 success = false;
338
339 // Report the name of not loaded files:
340 while( ii < aFilenameList.GetCount() )
341 {
342 filename = aFilenameList[ii++];
343 wxString txt = wxString::Format( MSG_NOT_LOADED, filename.GetFullName() );
344 reporter.Report( txt, RPT_SEVERITY_ERROR );
345 }
346 break;
347 }
348
349 SetActiveLayer( layer, false );
350 visibility[ layer ] = true;
351
352 try
353 {
354 // 2 = Autodetect
355 if( ( *aFileType )[ii] == 2 )
356 {
357 if( EXCELLON_IMAGE::TestFileIsExcellon( filename.GetFullPath() ) )
358 ( *aFileType )[ii] = 1;
359 else if( GERBER_FILE_IMAGE::TestFileIsRS274( filename.GetFullPath() ) )
360 ( *aFileType )[ii] = 0;
361 }
362
363 switch( ( *aFileType )[ii] )
364 {
365 case 0:
366
367 if( Read_GERBER_File( filename.GetFullPath() ) )
368 {
369 UpdateFileHistory( filename.GetFullPath() );
370
371 if( firstLoadedLayer == NO_AVAILABLE_LAYERS )
372 {
373 firstLoadedLayer = layer;
374 }
375 }
376
377 break;
378
379 case 1:
380
381 if( Read_EXCELLON_File( filename.GetFullPath() ) )
382 {
383 UpdateFileHistory( filename.GetFullPath(), &m_drillFileHistory );
384
385 // Select the first added layer by default when done loading
386 if( firstLoadedLayer == NO_AVAILABLE_LAYERS )
387 {
388 firstLoadedLayer = layer;
389 }
390 }
391
392 break;
393 default:
394 wxString txt = wxString::Format( MSG_NOT_LOADED, filename.GetFullName() );
395 reporter.Report( txt, RPT_SEVERITY_ERROR );
396 }
397 }
398 catch( const std::bad_alloc& )
399 {
400 wxString txt = wxString::Format( MSG_OOM, filename.GetFullName() );
401 reporter.Report( txt, RPT_SEVERITY_ERROR );
402 success = false;
403 continue;
404 }
405
406 if( progress )
407 progress->AdvanceProgress();
408 }
409
410 if( !success )
411 {
412 wxSafeYield(); // Allows slice of time to redraw the screen
413 // to refresh widgets, before displaying messages
414 HTML_MESSAGE_BOX mbox( this, _( "Errors" ) );
415 mbox.ListSet( reporter.GetMessages() );
416 mbox.ShowModal();
417 }
418
420
421 if( firstLoadedLayer != NO_AVAILABLE_LAYERS )
422 SetActiveLayer( firstLoadedLayer, true );
423
424 // Synchronize layers tools with actual active layer:
426
427 m_LayersManager->UpdateLayerIcons();
428 syncLayerBox( true );
429
430 GetCanvas()->Refresh();
431
432 return success;
433}
434
435
436bool GERBVIEW_FRAME::unarchiveFiles( const wxString& aFullFileName, REPORTER* aReporter )
437{
438 bool foundX2Gerbers = false;
439 wxString msg;
440 int firstLoadedLayer = NO_AVAILABLE_LAYERS;
442
443 // Extract the path of aFullFileName. We use it to store temporary files
444 wxFileName fn( aFullFileName );
445 wxString unzipDir = fn.GetPath();
446
447 wxFFileInputStream zipFile( aFullFileName );
448
449 if( !zipFile.IsOk() )
450 {
451 if( aReporter )
452 {
453 msg.Printf( _( "Zip file '%s' cannot be opened." ), aFullFileName );
454 aReporter->Report( msg, RPT_SEVERITY_ERROR );
455 }
456
457 return false;
458 }
459
460 // Update the list of recent zip files.
461 UpdateFileHistory( aFullFileName, &m_zipFileHistory );
462
463 // The unzipped file in only a temporary file. Give it a filename
464 // which cannot conflict with an usual filename.
465 // TODO: make Read_GERBER_File() and Read_EXCELLON_File() able to
466 // accept a stream, and avoid using a temp file.
467 wxFileName temp_fn( "$tempfile.tmp" );
468 temp_fn.MakeAbsolute( unzipDir );
469 wxString unzipped_tempfile = temp_fn.GetFullPath();
470
471
472 bool success = true;
473 wxZipInputStream zipArchive( zipFile );
474 wxZipEntry* entry;
475 bool reported_no_more_layer = false;
476 KIGFX::VIEW* view = GetCanvas()->GetView();
477
478 while( ( entry = zipArchive.GetNextEntry() ) != nullptr )
479 {
480 if( entry->IsDir() )
481 continue;
482
483 wxString fname = entry->GetName();
484 wxFileName uzfn = fname;
485 wxString curr_ext = uzfn.GetExt().Lower();
486
487 // The archive contains Gerber and/or Excellon drill files. Use the right loader.
488 // However it can contain a few other files (reports, pdf files...),
489 // which will be skipped.
490 if( curr_ext == FILEEXT::GerberJobFileExtension.c_str() )
491 {
492 //We cannot read a gerber job file as a gerber plot file: skip it
493 if( aReporter )
494 {
495 msg.Printf( _( "Skipped file '%s' (gerber job file)." ), entry->GetName() );
496 aReporter->Report( msg, RPT_SEVERITY_WARNING );
497 }
498
499 continue;
500 }
501
502 wxString matchedExt;
503 enum GERBER_ORDER_ENUM order;
504 GERBER_FILE_IMAGE_LIST::GetGerberLayerFromFilename( fname, order, matchedExt );
505
506 int layer = getNextAvailableLayer();
507
508 if( layer == NO_AVAILABLE_LAYERS )
509 {
510 success = false;
511
512 if( aReporter )
513 {
514 if( !reported_no_more_layer )
516
517 reported_no_more_layer = true;
518
519 // Report the name of not loaded files:
520 msg.Printf( MSG_NOT_LOADED, entry->GetName() );
521 aReporter->Report( msg, RPT_SEVERITY_ERROR );
522 }
523
524 delete entry;
525 continue;
526 }
527
528 SetActiveLayer( layer, false );
529
530 // Create the unzipped temporary file:
531 {
532 wxFFileOutputStream temporary_ofile( unzipped_tempfile );
533
534 if( temporary_ofile.Ok() )
535 temporary_ofile.Write( zipArchive );
536 else
537 {
538 success = false;
539
540 if( aReporter )
541 {
542 msg.Printf( _( "<b>Unable to create temporary file '%s'.</b>" ),
543 unzipped_tempfile );
544 aReporter->Report( msg, RPT_SEVERITY_ERROR );
545 }
546 }
547 }
548
549 bool read_ok = true;
550
551 // Try to parse files if we can't tell from file extension
553 {
554 if( EXCELLON_IMAGE::TestFileIsExcellon( unzipped_tempfile ) )
555 {
557 }
558 else if( GERBER_FILE_IMAGE::TestFileIsRS274( unzipped_tempfile ) )
559 {
560 // If we have no way to know what layer it is, just guess
562 }
563 else
564 {
565 if( aReporter )
566 {
567 msg.Printf( _( "Skipped file '%s' (unknown type)." ), entry->GetName() );
568 aReporter->Report( msg, RPT_SEVERITY_WARNING );
569 }
570 }
571 }
572
574 {
575 read_ok = Read_EXCELLON_File( unzipped_tempfile );
576 }
578 {
579 // Read gerber files: each file is loaded on a new GerbView layer
580 read_ok = Read_GERBER_File( unzipped_tempfile );
581
582 if( read_ok )
583 {
584 if( GERBER_FILE_IMAGE* gbrImage = GetGbrImage( layer ) )
585 view->SetLayerHasNegatives( GERBER_DRAW_LAYER( layer ), gbrImage->HasNegativeItems() );
586 }
587 }
588
589 // Select the first added layer by default when done loading
590 if( read_ok && firstLoadedLayer == NO_AVAILABLE_LAYERS )
591 {
592 firstLoadedLayer = layer;
593 }
594
595 delete entry;
596
597 // The unzipped file is only a temporary file, delete it.
598 wxRemoveFile( unzipped_tempfile );
599
600 if( !read_ok )
601 {
602 success = false;
603
604 if( aReporter )
605 {
606 msg.Printf( _( "<b>unzipped file %s read error</b>" ), unzipped_tempfile );
607 aReporter->Report( msg, RPT_SEVERITY_ERROR );
608 }
609 }
610 else
611 {
612 GERBER_FILE_IMAGE* gerber_image = GetGbrImage( layer );
613 visibility[ layer ] = true;
614
615 if( gerber_image )
616 {
617 gerber_image->m_FileName = fname;
618 if( gerber_image->m_IsX2_file )
619 foundX2Gerbers = true;
620 }
621
622 layer = getNextAvailableLayer();
623 SetActiveLayer( layer, false );
624 }
625 }
626
627 if( foundX2Gerbers )
629 else
631
633
634 // Select the first layer loaded so we don't show another layer on top after
635 if( firstLoadedLayer != NO_AVAILABLE_LAYERS )
636 SetActiveLayer( firstLoadedLayer, true );
637
638 return success;
639}
640
641
642bool GERBVIEW_FRAME::LoadZipArchiveFile( const wxString& aFullFileName )
643{
644#define ZipFileExtension "zip"
645
646 wxFileName filename = aFullFileName;
647 wxString currentPath;
648
649 if( !filename.IsOk() )
650 {
651 // Use the current working directory if the file name path does not exist.
652 if( filename.DirExists() )
653 currentPath = filename.GetPath();
654 else
655 currentPath = m_mruPath;
656
657 wxFileDialog dlg( this, _( "Open Zip File" ), currentPath, filename.GetFullName(),
659 wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_CHANGE_DIR );
660
662
663 if( dlg.ShowModal() == wxID_CANCEL )
664 return false;
665
666 filename = dlg.GetPath();
667 currentPath = wxGetCwd();
668 m_mruPath = currentPath;
669 }
670 else
671 {
672 currentPath = filename.GetPath();
673 m_mruPath = currentPath;
674 }
675
677
678 if( filename.IsOk() )
679 unarchiveFiles( filename.GetFullPath(), &reporter );
680
681 Zoom_Automatique( false );
682
683 // Synchronize layers tools with actual active layer:
686 m_LayersManager->UpdateLayerIcons();
687 syncLayerBox();
688
689 if( reporter.HasMessage() )
690 {
691 wxSafeYield(); // Allows slice of time to redraw the screen
692 // to refresh widgets, before displaying messages
693 HTML_MESSAGE_BOX mbox( this, _( "Messages" ) );
694 mbox.ListSet( reporter.GetMessages() );
695 mbox.ShowModal();
696 }
697
698 return true;
699}
700
701
703{
704 m_drillFileHistory.ClearFileHistory();
705 m_zipFileHistory.ClearFileHistory();
706 m_jobFileHistory.ClearFileHistory();
707
709}
710
711
713{
714 wxString gerbFn; // param to be sent with action event.
715
716 for( const wxFileName& file : m_AcceptedFiles )
717 {
718 if( file.GetExt() == FILEEXT::ArchiveFileExtension )
719 {
720 wxString fn = file.GetFullPath();
721 // Open zip archive in editor
722 m_toolManager->RunAction<wxString*>( *m_acceptedExts.at( FILEEXT::ArchiveFileExtension ), &fn );
723 }
724 else
725 {
726 // Store FileName in variable to open later
727 gerbFn += '"' + file.GetFullPath() + '"';
728 }
729 }
730
731 // Open files in editor
732 if( !gerbFn.IsEmpty() )
733 m_toolManager->RunAction<wxString*>( *m_acceptedExts.at( FILEEXT::GerberFileExtension ), &gerbFn );
734}
int ShowModal() override
std::vector< wxFileName > m_AcceptedFiles
void UpdateFileHistory(const wxString &FullFileName, FILE_HISTORY *aFileHistory=nullptr)
Update the list of recently opened files.
virtual void ClearFileHistory()
Remove all files from the file history.
std::map< const wxString, TOOL_ACTION * > m_acceptedExts
Associate file extensions with action to execute.
wxString GetFileFromHistory(int cmdId, const wxString &type, FILE_HISTORY *aFileHistory=nullptr)
Fetch the file name from the file history list.
void ReCreateMenuBar()
Recreate the menu bar.
virtual void Zoom_Automatique(bool aWarpPointer)
Redraw the screen with best zoom level and the best centering that shows all the page or the board.
virtual EDA_DRAW_PANEL_GAL * GetCanvas() const
Return a pointer to GAL-based canvas of given EDA draw frame.
virtual KIGFX::VIEW * GetView() const
Return a pointer to the #VIEW instance used in the panel.
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
static bool TestFileIsExcellon(const wxString &aFullFileName)
Performs a heuristics-based check of whether the file is an Excellon drill file.
unsigned GetLoadedImageCount()
Get number of loaded images.
static void GetGerberLayerFromFilename(const wxString &filename, enum GERBER_ORDER_ENUM &order, wxString &matchedExtension)
Utility function to guess which PCB layer of a gerber/drill file corresponds to based on its file ext...
Hold the image data and parameters for one gerber file and layer parameters.
wxString m_FileName
Full File Name for this layer.
static bool TestFileIsRS274(const wxString &aFullFileName)
Performs a heuristics-based check of whether the file is an RS274 gerber file.
Definition readgerb.cpp:134
bool m_IsX2_file
True if a X2 gerber attribute was found in file.
void OnDrlFileHistory(wxCommandEvent &event)
Delete the current data and load a drill file in Excellon format selected from history list on curren...
void SortLayersByX2Attributes()
LSET GetVisibleLayers() const
A proxy function that calls the correspondent function in m_BoardSettings.
bool Read_EXCELLON_File(const wxString &aFullFileName)
bool LoadFileOrShowDialog(const wxString &aFileName, const wxString &dialogFiletypes, const wxString &dialogTitle, const int filetype)
Loads the file provided or shows a dialog to get the file(s) from the user.
bool LoadGerberJobFile(const wxString &aFileName)
Load a Gerber job file, and load gerber files found in job files.
void OnClearDrlFileHistory(wxCommandEvent &aEvent)
GERBER_FILE_IMAGE_LIST * GetImagesList() const
Accessors to GERBER_FILE_IMAGE_LIST and GERBER_FILE_IMAGE data.
wxString m_lastFileName
void syncLayerBox(bool aRebuildLayerBox=false)
Update the currently "selected" layer within m_SelLayerBox.
bool LoadGerberFiles(const wxString &aFileName)
Load a given Gerber file or selected file(s), if the filename is empty.
bool unarchiveFiles(const wxString &aFullFileName, REPORTER *aReporter=nullptr)
Extract gerber and drill files from the zip archive, and load them.
FILE_HISTORY m_jobFileHistory
void OnJobFileHistory(wxCommandEvent &event)
Delete the current data and load a gerber job file selected from the history list.
void OnZipFileHistory(wxCommandEvent &event)
Delete the current data and load a zip archive file selected from the history list.
int GetActiveLayer() const
Return the active layer.
GERBER_LAYER_WIDGET * m_LayersManager
void SetActiveLayer(int aLayer, bool doLayerWidgetUpdate=true)
change the currently active layer to aLayer and update the GERBER_LAYER_WIDGET.
void SetVisibleLayers(const LSET &aLayerMask)
A proxy function that calls the correspondent function in m_BoardSettings.
void ClearFileHistory() override
Remove all files from the file history.
bool Read_GERBER_File(const wxString &GERBER_FullFileName)
Definition readgerb.cpp:37
void SortLayersByFileExtension()
GERBER_FILE_IMAGE * GetGbrImage(int aIdx) const
bool LoadListOfGerberAndDrillFiles(const wxString &aPath, const wxArrayString &aFilenameList, std::vector< int > *aFileType)
Load a list of Gerber and NC drill files and updates the view based on them.
bool LoadAutodetectedFiles(const wxString &aFileName)
Load a given file or selected file(s), if the filename is empty.
void ReFillLayerWidget()
Change out all the layers in m_Layers; called upon loading new gerber files.
FILE_HISTORY m_zipFileHistory
int getNextAvailableLayer() const
Find the next empty layer.
bool LoadZipArchiveFile(const wxString &aFileName)
Load a zipped archive file.
void OnClearGbrFileHistory(wxCommandEvent &aEvent)
void OnClearZipFileHistory(wxCommandEvent &aEvent)
void DoWithAcceptedFiles() override
Execute action on accepted dropped file.
void OnGbrFileHistory(wxCommandEvent &event)
Delete the current data and loads a Gerber file selected from history list on current layer.
FILE_HISTORY m_drillFileHistory
bool LoadExcellonFiles(const wxString &aFileName)
Load a drill (EXCELLON) file or many files.
void OnClearJobFileHistory(wxCommandEvent &aEvent)
void ListSet(const wxString &aList)
Add a list of items.
Hold a (potentially large) number of VIEW_ITEMs and renders them on a graphics device provided by the...
Definition view.h:63
void SetLayerHasNegatives(int aLayer, bool aNegatives=true)
Set the status of negatives presense in a particular layer.
Definition view.h:471
LSET is a set of PCB_LAYER_IDs.
Definition lset.h:37
A pure virtual class used to derive REPORTER objects from.
Definition reporter.h:71
virtual REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED)
Report a string with a given severity.
Definition reporter.h:100
TOOL_MANAGER * m_toolManager
A wrapper for reporting to a wxString object.
Definition reporter.h:189
#define _(s)
#define MSG_NOT_LOADED
#define MSG_OOM
#define MSG_NO_MORE_LAYER
#define NO_AVAILABLE_LAYERS
static const std::string GerberJobFileExtension
static const std::string GerberFileExtension
static const std::string ArchiveFileExtension
static wxString AllFilesWildcard()
static wxString DrillFileWildcard()
static wxString ZipFileWildcard()
#define GERBER_DRAW_LAYER(x)
Definition layer_ids.h:538
void AllowNetworkFileSystems(wxDialog *aDialog)
Configure a file dialog to show network and virtual file systems.
Definition wxgtk/ui.cpp:448
@ RPT_SEVERITY_WARNING
@ RPT_SEVERITY_ERROR
IbisParser parser & reporter
wxString AddFileExtListToFilter(const std::vector< std::string > &aExts)
Build the wildcard extension file dialog wildcard filter to add to the base message dialog.
Definition of file extensions used in Kicad.
#define PR_CAN_ABORT