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