KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcbnew_jobs_handler.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) 2022 Mark Roszko <mark.roszko@gmail.com>
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software: you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License as published by the
9 * Free Software Foundation, either version 3 of the License, or (at your
10 * option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License along
18 * with this program. If not, see <http://www.gnu.org/licenses/>.
19 */
20
21#include <wx/dir.h>
22#include "pcbnew_jobs_handler.h"
23#include <board_commit.h>
25#include <drc/drc_item.h>
26#include <drc/drc_report.h>
30#include <jobs/job_fp_upgrade.h>
43#include <jobs/job_pcb_render.h>
44#include <jobs/job_pcb_drc.h>
45#include <lset.h>
46#include <cli/exit_codes.h>
52#include <tool/tool_manager.h>
53#include <tools/drc_tool.h>
54#include <filename_resolver.h>
59#include <kiface_base.h>
60#include <macros.h>
61#include <pad.h>
62#include <pcb_marker.h>
66#include <kiface_ids.h>
69#include <pcbnew_settings.h>
70#include <pcbplot.h>
71#include <pcb_plotter.h>
72#include <pgm_base.h>
75#include <project_pcb.h>
77#include <reporter.h>
79#include <export_vrml.h>
80#include <wx/wfstream.h>
81#include <wx/zipstrm.h>
88#include <dialogs/dialog_plot.h>
92#include <paths.h>
93
95#include <locale_io.h>
96
97
98#ifdef _WIN32
99#ifdef TRANSPARENT
100#undef TRANSPARENT
101#endif
102#endif
103
104
106 JOB_DISPATCHER( aKiway ),
107 m_cliBoard( nullptr )
108{
109 Register( "3d", std::bind( &PCBNEW_JOBS_HANDLER::JobExportStep, this, std::placeholders::_1 ),
110 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
111 {
112 JOB_EXPORT_PCB_3D* svgJob = dynamic_cast<JOB_EXPORT_PCB_3D*>( job );
113
114 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
115 aKiway->Player( FRAME_PCB_EDITOR, false ) );
116
117 wxCHECK( svgJob && editFrame, false );
118
119 DIALOG_EXPORT_STEP dlg( editFrame, aParent, "", svgJob );
120 return dlg.ShowModal() == wxID_OK;
121 } );
122 Register( "render",
123 std::bind( &PCBNEW_JOBS_HANDLER::JobExportRender, this, std::placeholders::_1 ),
124 []( JOB* job, wxWindow* aParent ) -> bool
125 {
126 DIALOG_RENDER_JOB dlg( aParent, dynamic_cast<JOB_PCB_RENDER*>( job ) );
127 return dlg.ShowModal() == wxID_OK;
128 } );
129 Register( "svg", std::bind( &PCBNEW_JOBS_HANDLER::JobExportSvg, this, std::placeholders::_1 ),
130 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
131 {
132 JOB_EXPORT_PCB_SVG* svgJob = dynamic_cast<JOB_EXPORT_PCB_SVG*>( job );
133
134 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
135 aKiway->Player( FRAME_PCB_EDITOR, false ) );
136
137 wxCHECK( svgJob && editFrame, false );
138
139 DIALOG_PLOT dlg( editFrame, aParent, svgJob );
140 return dlg.ShowModal() == wxID_OK;
141 } );
142 Register( "gencad",
143 std::bind( &PCBNEW_JOBS_HANDLER::JobExportGencad, this, std::placeholders::_1 ),
144 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
145 {
146 JOB_EXPORT_PCB_GENCAD* gencadJob = dynamic_cast<JOB_EXPORT_PCB_GENCAD*>( job );
147
148 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
149 aKiway->Player( FRAME_PCB_EDITOR, false ) );
150
151 wxCHECK( gencadJob && editFrame, false );
152
153 DIALOG_GENCAD_EXPORT_OPTIONS dlg( editFrame, gencadJob );
154 return dlg.ShowModal() == wxID_OK;
155 } );
156 Register( "dxf", std::bind( &PCBNEW_JOBS_HANDLER::JobExportDxf, this, std::placeholders::_1 ),
157 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
158 {
159 JOB_EXPORT_PCB_DXF* dxfJob = dynamic_cast<JOB_EXPORT_PCB_DXF*>( job );
160
161 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
162 aKiway->Player( FRAME_PCB_EDITOR, false ) );
163
164 wxCHECK( dxfJob && editFrame, false );
165
166 DIALOG_PLOT dlg( editFrame, aParent, dxfJob );
167 return dlg.ShowModal() == wxID_OK;
168 } );
169 Register( "pdf", std::bind( &PCBNEW_JOBS_HANDLER::JobExportPdf, this, std::placeholders::_1 ),
170 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
171 {
172 JOB_EXPORT_PCB_PDF* pdfJob = dynamic_cast<JOB_EXPORT_PCB_PDF*>( job );
173
174 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
175 aKiway->Player( FRAME_PCB_EDITOR, false ) );
176
177 wxCHECK( pdfJob && editFrame, false );
178
179 DIALOG_PLOT dlg( editFrame, aParent, pdfJob );
180 return dlg.ShowModal() == wxID_OK;
181 } );
182 Register( "gerber",
183 std::bind( &PCBNEW_JOBS_HANDLER::JobExportGerber, this, std::placeholders::_1 ),
184 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
185 {
186 JOB_EXPORT_PCB_GERBER* gJob = dynamic_cast<JOB_EXPORT_PCB_GERBER*>( job );
187
188 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
189 aKiway->Player( FRAME_PCB_EDITOR, false ) );
190
191 wxCHECK( gJob && editFrame, false );
192
193 DIALOG_PLOT dlg( editFrame, aParent, gJob );
194 return dlg.ShowModal() == wxID_OK;
195 } );
196 Register( "gerbers",
197 std::bind( &PCBNEW_JOBS_HANDLER::JobExportGerbers, this, std::placeholders::_1 ),
198 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
199 {
200 JOB_EXPORT_PCB_GERBERS* gJob = dynamic_cast<JOB_EXPORT_PCB_GERBERS*>( job );
201
202 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
203 aKiway->Player( FRAME_PCB_EDITOR, false ) );
204
205 wxCHECK( gJob && editFrame, false );
206
207 DIALOG_PLOT dlg( editFrame, aParent, gJob );
208 return dlg.ShowModal() == wxID_OK;
209 } );
210 Register( "drill",
211 std::bind( &PCBNEW_JOBS_HANDLER::JobExportDrill, this, std::placeholders::_1 ),
212 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
213 {
214 JOB_EXPORT_PCB_DRILL* drillJob = dynamic_cast<JOB_EXPORT_PCB_DRILL*>( job );
215
216 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
217 aKiway->Player( FRAME_PCB_EDITOR, false ) );
218
219 wxCHECK( drillJob && editFrame, false );
220
221 DIALOG_GENDRILL dlg( editFrame, drillJob, aParent );
222 return dlg.ShowModal() == wxID_OK;
223 } );
224 Register( "pos", std::bind( &PCBNEW_JOBS_HANDLER::JobExportPos, this, std::placeholders::_1 ),
225 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
226 {
227 JOB_EXPORT_PCB_POS* posJob = dynamic_cast<JOB_EXPORT_PCB_POS*>( job );
228
229 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
230 aKiway->Player( FRAME_PCB_EDITOR, false ) );
231
232 wxCHECK( posJob && editFrame, false );
233
234 DIALOG_GEN_FOOTPRINT_POSITION dlg( posJob, editFrame, aParent );
235 return dlg.ShowModal() == wxID_OK;
236 } );
237 Register( "fpupgrade",
238 std::bind( &PCBNEW_JOBS_HANDLER::JobExportFpUpgrade, this, std::placeholders::_1 ),
239 []( JOB* job, wxWindow* aParent ) -> bool
240 {
241 return true;
242 } );
243 Register( "fpsvg",
244 std::bind( &PCBNEW_JOBS_HANDLER::JobExportFpSvg, this, std::placeholders::_1 ),
245 []( JOB* job, wxWindow* aParent ) -> bool
246 {
247 return true;
248 } );
249 Register( "drc", std::bind( &PCBNEW_JOBS_HANDLER::JobExportDrc, this, std::placeholders::_1 ),
250 []( JOB* job, wxWindow* aParent ) -> bool
251 {
252 DIALOG_DRC_JOB_CONFIG dlg( aParent, dynamic_cast<JOB_PCB_DRC*>( job ) );
253
254 return dlg.ShowModal() == wxID_OK;
255 } );
256 Register( "ipc2581",
257 std::bind( &PCBNEW_JOBS_HANDLER::JobExportIpc2581, this, std::placeholders::_1 ),
258 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
259 {
260 JOB_EXPORT_PCB_IPC2581* ipcJob = dynamic_cast<JOB_EXPORT_PCB_IPC2581*>( job );
261
262 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
263 aKiway->Player( FRAME_PCB_EDITOR, false ) );
264
265 wxCHECK( ipcJob && editFrame, false );
266
267 DIALOG_EXPORT_2581 dlg( ipcJob, editFrame, aParent );
268 return dlg.ShowModal() == wxID_OK;
269 } );
270 Register( "ipcd356",
271 std::bind( &PCBNEW_JOBS_HANDLER::JobExportIpcD356, this, std::placeholders::_1 ),
272 []( JOB* job, wxWindow* aParent ) -> bool
273 {
274 return true;
275 } );
276 Register( "odb",
277 std::bind( &PCBNEW_JOBS_HANDLER::JobExportOdb, this, std::placeholders::_1 ),
278 [aKiway]( JOB* job, wxWindow* aParent ) -> bool
279 {
280 JOB_EXPORT_PCB_ODB* odbJob = dynamic_cast<JOB_EXPORT_PCB_ODB*>( job );
281
282 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>(
283 aKiway->Player( FRAME_PCB_EDITOR, false ) );
284
285 wxCHECK( odbJob && editFrame, false );
286
287 DIALOG_EXPORT_ODBPP dlg( odbJob, editFrame, aParent );
288 return dlg.ShowModal() == wxID_OK;
289 } );
290}
291
292
293BOARD* PCBNEW_JOBS_HANDLER::getBoard( const wxString& aPath )
294{
295 BOARD* brd = nullptr;
296
297 if( !Pgm().IsGUI() && Pgm().GetSettingsManager().IsProjectOpen() )
298 {
299 wxString pcbPath = aPath;
300
301 if( pcbPath.IsEmpty() )
302 {
303 wxFileName path = Pgm().GetSettingsManager().Prj().GetProjectFullName();
305 path.MakeAbsolute();
306 pcbPath = path.GetFullPath();
307 }
308
309 if( !m_cliBoard )
310 m_cliBoard = LoadBoard( pcbPath, true );
311
312 brd = m_cliBoard;
313 }
314 else if( Pgm().IsGUI() && Pgm().GetSettingsManager().IsProjectOpen() )
315 {
317
318 if( editFrame )
319 brd = editFrame->GetBoard();
320 }
321 else
322 {
323 brd = LoadBoard( aPath, true );
324 }
325
326 if( !brd )
327 m_reporter->Report( _( "Failed to load board\n" ), RPT_SEVERITY_ERROR );
328
329 return brd;
330}
331
332
333LSEQ PCBNEW_JOBS_HANDLER::convertLayerArg( wxString& aLayerString, BOARD* aBoard ) const
334{
335 std::map<wxString, LSET> layerMasks;
336 std::map<wxString, LSET> layerGuiMasks;
337
338 // Build list of layer names and their layer mask:
339 for( PCB_LAYER_ID layer : LSET::AllLayersMask().Seq() )
340 {
341 // Add layer name used in pcb files
342 layerMasks[ LSET::Name( layer ) ] = LSET( { layer } );
343 // Add layer name using GUI canonical layer name
344 layerGuiMasks[ LayerName( layer ) ] = LSET( { layer } );
345
346 // Add user layer name
347 if( aBoard )
348 layerGuiMasks[ aBoard->GetLayerName( layer ) ] = LSET( { layer } );
349 }
350
351 // Add list of grouped layer names used in pcb files
352 layerMasks[ wxT( "*" ) ] = LSET::AllLayersMask();
353 layerMasks[ wxT( "*.Cu" ) ] = LSET::AllCuMask();
354 layerMasks[ wxT( "*In.Cu" ) ] = LSET::InternalCuMask();
355 layerMasks[ wxT( "F&B.Cu" ) ] = LSET( { F_Cu, B_Cu } );
356 layerMasks[ wxT( "*.Adhes" ) ] = LSET( { B_Adhes, F_Adhes } );
357 layerMasks[ wxT( "*.Paste" ) ] = LSET( { B_Paste, F_Paste } );
358 layerMasks[ wxT( "*.Mask" ) ] = LSET( { B_Mask, F_Mask } );
359 layerMasks[ wxT( "*.SilkS" ) ] = LSET( { B_SilkS, F_SilkS } );
360 layerMasks[ wxT( "*.Fab" ) ] = LSET( { B_Fab, F_Fab } );
361 layerMasks[ wxT( "*.CrtYd" ) ] = LSET( { B_CrtYd, F_CrtYd } );
362
363 // Add list of grouped layer names using GUI canonical layer names
364 layerGuiMasks[ wxT( "*.Adhesive" ) ] = LSET( { B_Adhes, F_Adhes } );
365 layerGuiMasks[ wxT( "*.Silkscreen" ) ] = LSET( { B_SilkS, F_SilkS } );
366 layerGuiMasks[ wxT( "*.Courtyard" ) ] = LSET( { B_CrtYd, F_CrtYd } );
367
368 LSEQ layerMask;
369
370 if( !aLayerString.IsEmpty() )
371 {
372 wxStringTokenizer layerTokens( aLayerString, "," );
373
374 while( layerTokens.HasMoreTokens() )
375 {
376 std::string token = TO_UTF8( layerTokens.GetNextToken() );
377
378 // Search for a layer name in canonical layer name used in .kicad_pcb files:
379 if( layerMasks.count( token ) )
380 {
381 for( PCB_LAYER_ID layer : layerMasks.at( token ).Seq() )
382 layerMask.push_back( layer );
383 }
384 // Search for a layer name in canonical layer name used in GUI (not translated):
385 else if( layerGuiMasks.count( token ) )
386 {
387 for( PCB_LAYER_ID layer : layerGuiMasks.at( token ).Seq() )
388 layerMask.push_back( layer );
389 }
390 else
391 {
392 m_reporter->Report( wxString::Format( _( "Invalid layer name \"%s\"\n" ),
393 token ) );
394 }
395 }
396 }
397
398 return layerMask;
399}
400
401
403{
404 JOB_EXPORT_PCB_3D* aStepJob = dynamic_cast<JOB_EXPORT_PCB_3D*>( aJob );
405
406 if( aStepJob == nullptr )
408
409 BOARD* brd = getBoard( aStepJob->m_filename );
410
411 if( !brd )
413
414 aJob->SetTitleBlock( brd->GetTitleBlock() );
415 brd->GetProject()->ApplyTextVars( aJob->GetVarOverrides() );
417
418 if( aStepJob->GetConfiguredOutputPath().IsEmpty() )
419 {
420 wxFileName fn = brd->GetFileName();
421 fn.SetName( fn.GetName() );
422
423 switch( aStepJob->m_format )
424 {
432 default:
433 m_reporter->Report( _( "Unknown export format" ), RPT_SEVERITY_ERROR );
434 return CLI::EXIT_CODES::ERR_UNKNOWN; // shouldnt have gotten here
435 }
436
437 aStepJob->SetWorkingOutputPath( fn.GetFullName() );
438 }
439
440 wxString outPath = aStepJob->GetFullOutputPath( brd->GetProject() );
441
442 if( !PATHS::EnsurePathExists( outPath, true ) )
443 {
444 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
446 }
447
449 {
450
451 double scale = 0.0;
452 switch ( aStepJob->m_vrmlUnits )
453 {
454 case JOB_EXPORT_PCB_3D::VRML_UNITS::MM: scale = 1.0; break;
455 case JOB_EXPORT_PCB_3D::VRML_UNITS::METERS: scale = 0.001; break;
456 case JOB_EXPORT_PCB_3D::VRML_UNITS::TENTHS: scale = 10.0 / 25.4; break;
457 case JOB_EXPORT_PCB_3D::VRML_UNITS::INCH: scale = 1.0 / 25.4; break;
458 }
459
460 EXPORTER_VRML vrmlExporter( brd );
461 wxString messages;
462
463 double originX = pcbIUScale.IUTomm( aStepJob->m_3dparams.m_Origin.x );
464 double originY = pcbIUScale.IUTomm( aStepJob->m_3dparams.m_Origin.y );
465
466 if( !aStepJob->m_hasUserOrigin )
467 {
468 BOX2I bbox = brd->ComputeBoundingBox( true );
469 originX = pcbIUScale.IUTomm( bbox.GetCenter().x );
470 originY = pcbIUScale.IUTomm( bbox.GetCenter().y );
471 }
472
473 bool success = vrmlExporter.ExportVRML_File( brd->GetProject(),
474 &messages,
475 outPath,
476 scale,
478 aStepJob->m_3dparams.m_IncludeDNP,
479 !aStepJob->m_vrmlModelDir.IsEmpty(),
480 aStepJob->m_vrmlRelativePaths,
481 aStepJob->m_vrmlModelDir,
482 originX,
483 originY );
484
485 if ( success )
486 {
487 m_reporter->Report( wxString::Format( _( "Successfully exported VRML to %s" ),
488 outPath ),
490 }
491 else
492 {
493 m_reporter->Report( _( "Error exporting VRML" ), RPT_SEVERITY_ERROR );
495 }
496 }
497 else
498 {
499 EXPORTER_STEP_PARAMS params = aStepJob->m_3dparams;
500
501 switch( aStepJob->m_format )
502 {
509 default:
510 m_reporter->Report( _( "Unknown export format" ), RPT_SEVERITY_ERROR );
511 return CLI::EXIT_CODES::ERR_UNKNOWN; // shouldnt have gotten here
512 }
513
514 EXPORTER_STEP stepExporter( brd, params );
515 stepExporter.m_outputFile = aStepJob->GetFullOutputPath( brd->GetProject() );
516
517 if( !stepExporter.Export() )
519 }
520
521 return CLI::EXIT_CODES::OK;
522}
523
524
526{
527 JOB_PCB_RENDER* aRenderJob = dynamic_cast<JOB_PCB_RENDER*>( aJob );
528
529 if( aRenderJob == nullptr )
531
532 // Reject width and height being invalid
533 // Final bit of sanity because this can blow things up
534 if( aRenderJob->m_width <= 0 || aRenderJob->m_height <= 0 )
535 {
536 m_reporter->Report( _( "Invalid image dimensions" ), RPT_SEVERITY_ERROR );
538 }
539
540 BOARD* brd = getBoard( aRenderJob->m_filename );
541
542 if( !brd )
544
545 aJob->SetTitleBlock( brd->GetTitleBlock() );
546 brd->GetProject()->ApplyTextVars( aJob->GetVarOverrides() );
548
549 if( aRenderJob->GetConfiguredOutputPath().IsEmpty() )
550 {
551 wxFileName fn = brd->GetFileName();
552
553 switch( aRenderJob->m_format )
554 {
557 default:
558 m_reporter->Report( _( "Unknown export format" ), RPT_SEVERITY_ERROR );
559 return CLI::EXIT_CODES::ERR_UNKNOWN; // shouldnt have gotten here
560 }
561
562 // set the name to board name + "side", its lazy but its hard to generate anything truely unique
563 // incase someone is doing this in a jobset with multiple jobs, they should be setting the output themselves
564 // or we do a hash based on all the options
565 fn.SetName( wxString::Format( "%s-%d", fn.GetName(), static_cast<int>( aRenderJob->m_side ) ) );
566
567 aRenderJob->SetWorkingOutputPath( fn.GetFullName() );
568 }
569
570 wxString outPath = aRenderJob->GetFullOutputPath( brd->GetProject() );
571
572 if( !PATHS::EnsurePathExists( outPath, true ) )
573 {
574 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
576 }
577
578 BOARD_ADAPTER boardAdapter;
579
580 boardAdapter.SetBoard( brd );
581 boardAdapter.m_IsBoardView = false;
582 boardAdapter.m_IsPreviewer = true; // Force display 3D models, regardless of 3D viewer options
583
586
587 if( EDA_3D_VIEWER_SETTINGS* userCfg = mgr.GetAppSettings<EDA_3D_VIEWER_SETTINGS>( "3d_viewer" ) )
588 {
589 cfg.m_Render = userCfg->m_Render;
590 cfg.m_Camera = userCfg->m_Camera;
591 }
592
593 if( aRenderJob->m_quality == JOB_PCB_RENDER::QUALITY::BASIC )
594 {
595 // Silkscreen is pixelated without antialiasing
597
598 cfg.m_Render.raytrace_backfloor = aRenderJob->m_floor;
599 cfg.m_Render.raytrace_post_processing = aRenderJob->m_floor;
600
602 cfg.m_Render.raytrace_reflections = false;
603 cfg.m_Render.raytrace_shadows = aRenderJob->m_floor;
604
605 // Better colors
607
608 // Tracks below soldermask are not visible without refractions
611 }
612 else if( aRenderJob->m_quality == JOB_PCB_RENDER::QUALITY::HIGH )
613 {
615 cfg.m_Render.raytrace_backfloor = true;
619 cfg.m_Render.raytrace_shadows = true;
622 }
623 else if( aRenderJob->m_quality == JOB_PCB_RENDER::QUALITY::JOB_SETTINGS )
624 {
626 cfg.m_Render.raytrace_backfloor = aRenderJob->m_floor;
629 }
630
632 aRenderJob->m_lightTopIntensity.y,
633 aRenderJob->m_lightTopIntensity.z, 1.0 );
634
636 aRenderJob->m_lightBottomIntensity.y,
637 aRenderJob->m_lightBottomIntensity.z, 1.0 );
638
640 aRenderJob->m_lightCameraIntensity.y,
641 aRenderJob->m_lightCameraIntensity.z, 1.0 );
642
643 COLOR4D lightColor( aRenderJob->m_lightSideIntensity.x,
644 aRenderJob->m_lightSideIntensity.y,
645 aRenderJob->m_lightSideIntensity.z, 1.0 );
646
648 lightColor, lightColor, lightColor, lightColor,
649 lightColor, lightColor, lightColor, lightColor,
650 };
651
652 int sideElevation = aRenderJob->m_lightSideElevation;
653
655 sideElevation, sideElevation, sideElevation, sideElevation,
656 -sideElevation, -sideElevation, -sideElevation, -sideElevation,
657 };
658
660 45, 135, 225, 315, 45, 135, 225, 315,
661 };
662
663 cfg.m_CurrentPreset = aRenderJob->m_colorPreset;
664 boardAdapter.m_Cfg = &cfg;
665
668 && aRenderJob->m_format == JOB_PCB_RENDER::FORMAT::PNG ) )
669 {
670 boardAdapter.m_ColorOverrides[LAYER_3D_BACKGROUND_TOP] = COLOR4D( 1.0, 1.0, 1.0, 0.0 );
671 boardAdapter.m_ColorOverrides[LAYER_3D_BACKGROUND_BOTTOM] = COLOR4D( 1.0, 1.0, 1.0, 0.0 );
672 }
673
675
676 static std::map<JOB_PCB_RENDER::SIDE, VIEW3D_TYPE> s_viewCmdMap = {
677 { JOB_PCB_RENDER::SIDE::TOP, VIEW3D_TYPE::VIEW3D_TOP },
678 { JOB_PCB_RENDER::SIDE::BOTTOM, VIEW3D_TYPE::VIEW3D_BOTTOM },
679 { JOB_PCB_RENDER::SIDE::LEFT, VIEW3D_TYPE::VIEW3D_LEFT },
680 { JOB_PCB_RENDER::SIDE::RIGHT, VIEW3D_TYPE::VIEW3D_RIGHT },
681 { JOB_PCB_RENDER::SIDE::FRONT, VIEW3D_TYPE::VIEW3D_FRONT },
682 { JOB_PCB_RENDER::SIDE::BACK, VIEW3D_TYPE::VIEW3D_BACK },
683 };
684
685 PROJECTION_TYPE projection = aRenderJob->m_perspective ? PROJECTION_TYPE::PERSPECTIVE
686 : PROJECTION_TYPE::ORTHO;
687
688 wxSize windowSize( aRenderJob->m_width, aRenderJob->m_height );
689 TRACK_BALL camera( 2 * RANGE_SCALE_3D );
690
691 camera.SetProjection( projection );
692 camera.SetCurWindowSize( windowSize );
693
694 RENDER_3D_RAYTRACE_RAM raytrace( boardAdapter, camera );
695 raytrace.SetCurWindowSize( windowSize );
696
697 for( bool first = true; raytrace.Redraw( false, m_reporter, m_reporter ); first = false )
698 {
699 if( first )
700 {
701 const float cmTo3D = boardAdapter.BiuTo3dUnits() * pcbIUScale.mmToIU( 10.0 );
702
703 // First redraw resets lookat point to the board center, so set up the camera here
704 camera.ViewCommand_T1( s_viewCmdMap[aRenderJob->m_side] );
705
706 camera.SetLookAtPos_T1( camera.GetLookAtPos_T1() + SFVEC3F( aRenderJob->m_pivot.x,
707 aRenderJob->m_pivot.y,
708 aRenderJob->m_pivot.z ) * cmTo3D );
709
710 camera.Pan_T1( SFVEC3F( aRenderJob->m_pan.x, aRenderJob->m_pan.y, aRenderJob->m_pan.z ) );
711
712 camera.Zoom_T1( aRenderJob->m_zoom );
713
714 camera.RotateX_T1( DEG2RAD( aRenderJob->m_rotation.x ) );
715 camera.RotateY_T1( DEG2RAD( aRenderJob->m_rotation.y ) );
716 camera.RotateZ_T1( DEG2RAD( aRenderJob->m_rotation.z ) );
717
718 camera.Interpolate( 1.0f );
719 camera.SetT0_and_T1_current_T();
720 camera.ParametersChanged();
721 }
722 }
723
724 uint8_t* rgbaBuffer = raytrace.GetBuffer();
725 wxSize realSize = raytrace.GetRealBufferSize();
726 bool success = !!rgbaBuffer;
727
728 if( rgbaBuffer )
729 {
730 const unsigned int wxh = realSize.x * realSize.y;
731
732 unsigned char* rgbBuffer = (unsigned char*) malloc( wxh * 3 );
733 unsigned char* alphaBuffer = (unsigned char*) malloc( wxh );
734
735 unsigned char* rgbaPtr = rgbaBuffer;
736 unsigned char* rgbPtr = rgbBuffer;
737 unsigned char* alphaPtr = alphaBuffer;
738
739 for( int y = 0; y < realSize.y; y++ )
740 {
741 for( int x = 0; x < realSize.x; x++ )
742 {
743 rgbPtr[0] = rgbaPtr[0];
744 rgbPtr[1] = rgbaPtr[1];
745 rgbPtr[2] = rgbaPtr[2];
746 alphaPtr[0] = rgbaPtr[3];
747
748 rgbaPtr += 4;
749 rgbPtr += 3;
750 alphaPtr += 1;
751 }
752 }
753
754 wxImage image( realSize );
755 image.SetData( rgbBuffer );
756 image.SetAlpha( alphaBuffer );
757 image = image.Mirror( false );
758
759 image.SetOption( wxIMAGE_OPTION_QUALITY, 90 );
760 image.SaveFile( outPath, aRenderJob->m_format == JOB_PCB_RENDER::FORMAT::PNG ? wxBITMAP_TYPE_PNG
761 : wxBITMAP_TYPE_JPEG );
762 }
763
764 if( success )
765 {
766 m_reporter->Report( _( "Successfully created 3D render image" ) + wxS( "\n" ), RPT_SEVERITY_INFO );
767 return CLI::EXIT_CODES::OK;
768 }
769 else
770 {
771 m_reporter->Report( _( "Error creating 3D render image" ) + wxS( "\n" ), RPT_SEVERITY_ERROR );
773 }
774}
775
776
778{
779 JOB_EXPORT_PCB_SVG* aSvgJob = dynamic_cast<JOB_EXPORT_PCB_SVG*>( aJob );
780
781 if( aSvgJob == nullptr )
783
784 BOARD* brd = getBoard( aSvgJob->m_filename );
785
786 if( !brd )
788
789 aJob->SetTitleBlock( brd->GetTitleBlock() );
790
792 {
793 if( aSvgJob->GetConfiguredOutputPath().IsEmpty() )
794 {
795 wxFileName fn = brd->GetFileName();
796 fn.SetName( fn.GetName() );
797 fn.SetExt( GetDefaultPlotExtension( PLOT_FORMAT::SVG ) );
798
799 aSvgJob->SetWorkingOutputPath( fn.GetFullName() );
800 }
801 }
802
803 wxString outPath = aSvgJob->GetFullOutputPath( brd->GetProject() );
804
806 {
807 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
809 }
810
812 brd->GetProject()->ApplyTextVars( aJob->GetVarOverrides() );
814
815 if( aSvgJob->m_argLayers )
816 aSvgJob->m_plotLayerSequence = convertLayerArg( aSvgJob->m_argLayers.value(), brd );
817
818 if( aSvgJob->m_argCommonLayers )
819 aSvgJob->m_plotOnAllLayersSequence = convertLayerArg( aSvgJob->m_argCommonLayers.value(), brd );
820
821 if( aSvgJob->m_plotLayerSequence.size() < 1 )
822 {
823 m_reporter->Report( _( "At least one layer must be specified\n" ), RPT_SEVERITY_ERROR );
825 }
826
827 PCB_PLOT_PARAMS plotOpts;
828 PCB_PLOTTER::PlotJobToPlotOpts( plotOpts, aSvgJob, *m_reporter );
829
830 PCB_PLOTTER plotter( brd, m_reporter, plotOpts );
831
832 std::optional<wxString> layerName;
833 std::optional<wxString> sheetName;
834 std::optional<wxString> sheetPath;
835
837 {
838 if( aJob->GetVarOverrides().contains( wxT( "LAYER" ) ) )
839 layerName = aSvgJob->GetVarOverrides().at( wxT( "LAYER" ) );
840
841 if( aJob->GetVarOverrides().contains( wxT( "SHEETNAME" ) ) )
842 sheetName = aSvgJob->GetVarOverrides().at( wxT( "SHEETNAME" ) );
843
844 if( aJob->GetVarOverrides().contains( wxT( "SHEETPATH" ) ) )
845 sheetPath = aSvgJob->GetVarOverrides().at( wxT( "SHEETPATH" ) );
846 }
847
848 if( !plotter.Plot( outPath, aSvgJob->m_plotLayerSequence, aSvgJob->m_plotOnAllLayersSequence,
850 layerName, sheetName, sheetPath ) )
851 {
853 }
854
855 return CLI::EXIT_CODES::OK;
856}
857
858
860{
861 JOB_EXPORT_PCB_DXF* aDxfJob = dynamic_cast<JOB_EXPORT_PCB_DXF*>( aJob );
862
863 if( aDxfJob == nullptr )
865
866 BOARD* brd = getBoard( aDxfJob->m_filename );
867
868 if( !brd )
870
871 aJob->SetTitleBlock( brd->GetTitleBlock() );
873 brd->GetProject()->ApplyTextVars( aJob->GetVarOverrides() );
875
876 if( aDxfJob->m_argLayers )
877 aDxfJob->m_plotLayerSequence = convertLayerArg( aDxfJob->m_argLayers.value(), brd );
878
879 if( aDxfJob->m_argCommonLayers )
880 aDxfJob->m_plotOnAllLayersSequence = convertLayerArg( aDxfJob->m_argCommonLayers.value(), brd );
881
882 if( aDxfJob->m_plotLayerSequence.size() < 1 )
883 {
884 m_reporter->Report( _( "At least one layer must be specified\n" ), RPT_SEVERITY_ERROR );
886 }
887
889 {
890 if( aDxfJob->GetConfiguredOutputPath().IsEmpty() )
891 {
892 wxFileName fn = brd->GetFileName();
893 fn.SetName( fn.GetName() );
894 fn.SetExt( GetDefaultPlotExtension( PLOT_FORMAT::DXF ) );
895
896 aDxfJob->SetWorkingOutputPath( fn.GetFullName() );
897 }
898 }
899
900 wxString outPath = aDxfJob->GetFullOutputPath( brd->GetProject() );
901
903 {
904 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
906 }
907
908 PCB_PLOT_PARAMS plotOpts;
909 PCB_PLOTTER::PlotJobToPlotOpts( plotOpts, aDxfJob, *m_reporter);
910
911 PCB_PLOTTER plotter( brd, m_reporter, plotOpts );
912
913 std::optional<wxString> layerName;
914 std::optional<wxString> sheetName;
915 std::optional<wxString> sheetPath;
916
918 {
919 if( aJob->GetVarOverrides().contains( wxT( "LAYER" ) ) )
920 layerName = aDxfJob->GetVarOverrides().at( wxT( "LAYER" ) );
921
922 if( aJob->GetVarOverrides().contains( wxT( "SHEETNAME" ) ) )
923 sheetName = aDxfJob->GetVarOverrides().at( wxT( "SHEETNAME" ) );
924
925 if( aJob->GetVarOverrides().contains( wxT( "SHEETPATH" ) ) )
926 sheetPath = aDxfJob->GetVarOverrides().at( wxT( "SHEETPATH" ) );
927 }
928
929 if( !plotter.Plot( outPath, aDxfJob->m_plotLayerSequence, aDxfJob->m_plotOnAllLayersSequence,
931 layerName, sheetName, sheetPath ) )
932 {
934 }
935
936 return CLI::EXIT_CODES::OK;
937}
938
939
941{
942 bool plotAllLayersOneFile = false;
943 JOB_EXPORT_PCB_PDF* pdfJob = dynamic_cast<JOB_EXPORT_PCB_PDF*>( aJob );
944
945 if( pdfJob == nullptr )
947
948 BOARD* brd = getBoard( pdfJob->m_filename );
949
950 if( !brd )
952
953 pdfJob->SetTitleBlock( brd->GetTitleBlock() );
955 brd->GetProject()->ApplyTextVars( pdfJob->GetVarOverrides() );
957
958 if( pdfJob->m_argLayers )
959 pdfJob->m_plotLayerSequence = convertLayerArg( pdfJob->m_argLayers.value(), brd );
960
961 if( pdfJob->m_argCommonLayers )
962 pdfJob->m_plotOnAllLayersSequence = convertLayerArg( pdfJob->m_argCommonLayers.value(), brd );
963
965 plotAllLayersOneFile = true;
966
967 if( pdfJob->m_plotLayerSequence.size() < 1 )
968 {
969 m_reporter->Report( _( "At least one layer must be specified\n" ), RPT_SEVERITY_ERROR );
971 }
972
973 if( plotAllLayersOneFile && pdfJob->GetConfiguredOutputPath().IsEmpty() )
974 {
975 wxFileName fn = brd->GetFileName();
976 fn.SetName( fn.GetName() );
977 fn.SetExt( GetDefaultPlotExtension( PLOT_FORMAT::PDF ) );
978
979 pdfJob->SetWorkingOutputPath( fn.GetFullName() );
980 }
981
982 PCB_PLOT_PARAMS plotOpts;
983 PCB_PLOTTER::PlotJobToPlotOpts( plotOpts, pdfJob, *m_reporter );
984
985 // ensure this is set for this one gen mode
986 if( plotAllLayersOneFile )
987 plotOpts.m_PDFSingle = true;
988
989 PCB_PLOTTER pcbPlotter( brd, m_reporter, plotOpts );
990
991 wxString outPath = pdfJob->GetFullOutputPath( brd->GetProject() );
992
993 if( !PATHS::EnsurePathExists( outPath, plotAllLayersOneFile ) )
994 {
995 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
997 }
998
999 std::optional<wxString> layerName;
1000 std::optional<wxString> sheetName;
1001 std::optional<wxString> sheetPath;
1002
1003 if( plotAllLayersOneFile )
1004 {
1005 if( pdfJob->GetVarOverrides().contains( wxT( "LAYER" ) ) )
1006 layerName = pdfJob->GetVarOverrides().at( wxT( "LAYER" ) );
1007
1008 if( pdfJob->GetVarOverrides().contains( wxT( "SHEETNAME" ) ) )
1009 sheetName = pdfJob->GetVarOverrides().at( wxT( "SHEETNAME" ) );
1010
1011 if( pdfJob->GetVarOverrides().contains( wxT( "SHEETPATH" ) ) )
1012 sheetPath = pdfJob->GetVarOverrides().at( wxT( "SHEETPATH" ) );
1013 }
1014
1015
1017
1018 if( !pcbPlotter.Plot( outPath, pdfJob->m_plotLayerSequence,
1019 pdfJob->m_plotOnAllLayersSequence, false, plotAllLayersOneFile,
1020 layerName, sheetName, sheetPath ) )
1021 {
1023 }
1024
1025 return CLI::EXIT_CODES::OK;
1026}
1027
1028
1030{
1031 int exitCode = CLI::EXIT_CODES::OK;
1032 JOB_EXPORT_PCB_GERBERS* aGerberJob = dynamic_cast<JOB_EXPORT_PCB_GERBERS*>( aJob );
1033
1034 if( aGerberJob == nullptr )
1036
1037 BOARD* brd = getBoard( aGerberJob->m_filename );
1038
1039 if( !brd )
1041
1042 wxString outPath = aGerberJob->GetFullOutputPath( brd->GetProject() );
1043
1044 if( !PATHS::EnsurePathExists( outPath, false ) )
1045 {
1046 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
1048 }
1049
1050 aJob->SetTitleBlock( brd->GetTitleBlock() );
1051 loadOverrideDrawingSheet( brd, aGerberJob->m_drawingSheet );
1052 brd->GetProject()->ApplyTextVars( aJob->GetVarOverrides() );
1053 brd->SynchronizeProperties();
1054
1055 if( aGerberJob->m_argLayers )
1056 {
1057 if( !aGerberJob->m_argLayers.value().empty() )
1058 aGerberJob->m_plotLayerSequence = convertLayerArg( aGerberJob->m_argLayers.value(), nullptr );
1059 else
1061 }
1062
1063 if( aGerberJob->m_argCommonLayers )
1064 aGerberJob->m_plotOnAllLayersSequence = convertLayerArg( aGerberJob->m_argCommonLayers.value(), brd );
1065
1066 PCB_PLOT_PARAMS boardPlotOptions = brd->GetPlotOptions();
1067 GERBER_JOBFILE_WRITER jobfile_writer( brd );
1068
1069 wxString fileExt;
1070
1071 if( aGerberJob->m_useBoardPlotParams )
1072 {
1073 // The board plot options are saved with all copper layers enabled, even those that don't
1074 // exist in the current stackup. This is done so the layers are automatically enabled in the plot
1075 // dialog when the user enables them. We need to filter out these not-enabled layers here so
1076 // we don't plot 32 layers when we only have 4, etc.
1077 LSET plotLayers = ( boardPlotOptions.GetLayerSelection() & LSET::AllNonCuMask() )
1078 | ( brd->GetEnabledLayers() & LSET::AllCuMask() );
1079 aGerberJob->m_plotLayerSequence = plotLayers.SeqStackupForPlotting();
1080 aGerberJob->m_plotOnAllLayersSequence = boardPlotOptions.GetPlotOnAllLayersSequence();
1081 }
1082 else
1083 {
1084 // default to the board enabled layers
1085 if( aGerberJob->m_plotLayerSequence.empty() )
1087 }
1088
1089 // Ensure layers to plot are restricted to enabled layers of the board to plot
1090 LSET layersToPlot = LSET( { aGerberJob->m_plotLayerSequence } ) & brd->GetEnabledLayers();
1091
1092 for( PCB_LAYER_ID layer : layersToPlot.UIOrder() )
1093 {
1094 LSEQ plotSequence;
1095
1096 // Base layer always gets plotted first.
1097 plotSequence.push_back( layer );
1098
1099 // Now all the "include on all" layers
1100 for( PCB_LAYER_ID layer_all : aGerberJob->m_plotOnAllLayersSequence )
1101 {
1102 // Don't plot the same layer more than once;
1103 if( find( plotSequence.begin(), plotSequence.end(), layer_all ) != plotSequence.end() )
1104 continue;
1105
1106 plotSequence.push_back( layer_all );
1107 }
1108
1109 // Pick the basename from the board file
1110 wxFileName fn( brd->GetFileName() );
1111 wxString layerName = brd->GetLayerName( layer );
1112 wxString sheetName;
1113 wxString sheetPath;
1114 PCB_PLOT_PARAMS plotOpts;
1115
1116 if( aGerberJob->m_useBoardPlotParams )
1117 plotOpts = boardPlotOptions;
1118 else
1119 populateGerberPlotOptionsFromJob( plotOpts, aGerberJob );
1120
1121 if( plotOpts.GetUseGerberProtelExtensions() )
1122 fileExt = GetGerberProtelExtension( layer );
1123 else
1125
1126 BuildPlotFileName( &fn, outPath, layerName, fileExt );
1127 wxString fullname = fn.GetFullName();
1128
1129 jobfile_writer.AddGbrFile( layer, fullname );
1130
1131 if( aJob->GetVarOverrides().contains( wxT( "LAYER" ) ) )
1132 layerName = aJob->GetVarOverrides().at( wxT( "LAYER" ) );
1133
1134 if( aJob->GetVarOverrides().contains( wxT( "SHEETNAME" ) ) )
1135 sheetName = aJob->GetVarOverrides().at( wxT( "SHEETNAME" ) );
1136
1137 if( aJob->GetVarOverrides().contains( wxT( "SHEETPATH" ) ) )
1138 sheetPath = aJob->GetVarOverrides().at( wxT( "SHEETPATH" ) );
1139
1140 // We are feeding it one layer at the start here to silence a logic check
1141 GERBER_PLOTTER* plotter;
1142 {
1144 plotter = (GERBER_PLOTTER*) StartPlotBoard( brd, &plotOpts, layer, layerName,
1145 fn.GetFullPath(), sheetName, sheetPath );
1146 }
1147
1148 if( plotter )
1149 {
1150 m_reporter->Report( wxString::Format( _( "Plotted to '%s'.\n" ), fn.GetFullPath() ),
1153 PlotBoardLayers( brd, plotter, plotSequence, plotOpts );
1154 plotter->EndPlot();
1155 }
1156 else
1157 {
1158 m_reporter->Report( wxString::Format( _( "Failed to plot to '%s'.\n" ),
1159 fn.GetFullPath() ),
1162 }
1163
1164 delete plotter;
1165 }
1166
1167 if( aGerberJob->m_createJobsFile )
1168 {
1169 wxFileName fn( brd->GetFileName() );
1170
1171 // Build gerber job file from basename
1172 BuildPlotFileName( &fn, outPath, wxT( "job" ), FILEEXT::GerberJobFileExtension );
1173 jobfile_writer.CreateJobFile( fn.GetFullPath() );
1174 }
1175
1176 return exitCode;
1177}
1178
1180{
1181 JOB_EXPORT_PCB_GENCAD* aGencadJob = dynamic_cast<JOB_EXPORT_PCB_GENCAD*>( aJob );
1182
1183 if( aGencadJob == nullptr )
1185
1186 BOARD* brd = LoadBoard( aGencadJob->m_filename, true ); // Ensure m_board is of type BOARD*
1187
1188 if( brd == nullptr )
1190
1191 GENCAD_EXPORTER exporter( brd );
1192
1193 VECTOR2I GencadOffset;
1194 VECTOR2I auxOrigin = brd->GetDesignSettings().GetAuxOrigin();
1195 GencadOffset.x = aGencadJob->m_useDrillOrigin ? auxOrigin.x : 0;
1196 GencadOffset.y = aGencadJob->m_useDrillOrigin ? auxOrigin.y : 0;
1197
1198 exporter.FlipBottomPads( aGencadJob->m_flipBottomPads );
1199 exporter.UsePinNamesUnique( aGencadJob->m_useUniquePins );
1200 exporter.UseIndividualShapes( aGencadJob->m_useIndividualShapes );
1201 exporter.SetPlotOffet( GencadOffset );
1202 exporter.StoreOriginCoordsInFile( aGencadJob->m_storeOriginCoords );
1203
1204 if( aGencadJob->GetConfiguredOutputPath().IsEmpty() )
1205 {
1206 wxFileName fn = brd->GetFileName();
1207 fn.SetName( fn.GetName() );
1208 fn.SetExt( FILEEXT::GencadFileExtension );
1209
1210 aGencadJob->SetWorkingOutputPath( fn.GetFullName() );
1211 }
1212
1213 wxString outPath = aGencadJob->GetFullOutputPath( brd->GetProject() );
1214
1215 if( !PATHS::EnsurePathExists( outPath, true ) )
1216 {
1217 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
1219 }
1220
1221 if( !exporter.WriteFile( outPath ) )
1222 {
1223 m_reporter->Report( wxString::Format( _( "Failed to create file '%s'.\n" ), outPath ),
1225
1227 }
1228
1229 m_reporter->Report( _( "Successfully created genCAD file\n" ), RPT_SEVERITY_INFO );
1230
1231 return CLI::EXIT_CODES::OK;
1232}
1233
1234
1236 JOB_EXPORT_PCB_GERBER* aJob )
1237{
1238 aPlotOpts.SetFormat( PLOT_FORMAT::GERBER );
1239
1240 aPlotOpts.SetPlotFrameRef( aJob->m_plotDrawingSheet );
1241 aPlotOpts.SetPlotValue( aJob->m_plotFootprintValues );
1242 aPlotOpts.SetPlotReference( aJob->m_plotRefDes );
1243
1245
1246 // Always disable plot pad holes
1247 aPlotOpts.SetDrillMarksType( DRILL_MARKS::NO_DRILL_SHAPE );
1248
1250 aPlotOpts.SetUseGerberX2format( aJob->m_useX2Format );
1252 aPlotOpts.SetUseAuxOrigin( aJob->m_useDrillOrigin );
1254 aPlotOpts.SetGerberPrecision( aJob->m_precision );
1255}
1256
1257
1260{
1261 populateGerberPlotOptionsFromJob( aPlotOpts, static_cast<JOB_EXPORT_PCB_GERBER*>( aJob ) );
1262
1263 aPlotOpts.SetCreateGerberJobFile( aJob->m_createJobsFile );
1264}
1265
1266
1268{
1269 int exitCode = CLI::EXIT_CODES::OK;
1270 JOB_EXPORT_PCB_GERBER* aGerberJob = dynamic_cast<JOB_EXPORT_PCB_GERBER*>( aJob );
1271
1272 if( aGerberJob == nullptr )
1274
1275 BOARD* brd = getBoard( aGerberJob->m_filename );
1276
1277 if( !brd )
1279
1280 aJob->SetTitleBlock( brd->GetTitleBlock() );
1281 brd->GetProject()->ApplyTextVars( aJob->GetVarOverrides() );
1282 brd->SynchronizeProperties();
1283
1284 if( aGerberJob->m_argLayers )
1285 aGerberJob->m_plotLayerSequence = convertLayerArg( aGerberJob->m_argLayers.value(), brd );
1286
1287 if( aGerberJob->m_argCommonLayers )
1288 aGerberJob->m_plotOnAllLayersSequence = convertLayerArg( aGerberJob->m_argCommonLayers.value(), brd );
1289
1290 if( aGerberJob->m_plotLayerSequence.size() < 1 )
1291 {
1292 m_reporter->Report( _( "At least one layer must be specified\n" ), RPT_SEVERITY_ERROR );
1294 }
1295
1296 if( aGerberJob->GetConfiguredOutputPath().IsEmpty() )
1297 {
1298 wxFileName fn = brd->GetFileName();
1299 fn.SetName( fn.GetName() );
1300 fn.SetExt( GetDefaultPlotExtension( PLOT_FORMAT::GERBER ) );
1301
1302 aGerberJob->SetWorkingOutputPath( fn.GetFullName() );
1303 }
1304
1305 PCB_PLOT_PARAMS plotOpts;
1306 populateGerberPlotOptionsFromJob( plotOpts, aGerberJob );
1307 plotOpts.SetLayerSelection( aGerberJob->m_plotLayerSequence );
1309
1311 wxString layerName;
1312 wxString sheetName;
1313 wxString sheetPath;
1314 wxString outPath = aGerberJob->GetFullOutputPath( brd->GetProject() );
1315
1316 // The first layer will be treated as the layer name for the gerber header,
1317 // the other layers will be treated equivalent to the "Plot on All Layers" option
1318 // in the GUI
1319 if( aGerberJob->m_plotLayerSequence.size() >= 1 )
1320 {
1321 layer = aGerberJob->m_plotLayerSequence.front();
1322 layerName = brd->GetLayerName( layer );
1323 }
1324
1325 if( aJob->GetVarOverrides().contains( wxT( "LAYER" ) ) )
1326 layerName = aJob->GetVarOverrides().at( wxT( "LAYER" ) );
1327
1328 if( aJob->GetVarOverrides().contains( wxT( "SHEETNAME" ) ) )
1329 sheetName = aJob->GetVarOverrides().at( wxT( "SHEETNAME" ) );
1330
1331 if( aJob->GetVarOverrides().contains( wxT( "SHEETPATH" ) ) )
1332 sheetPath = aJob->GetVarOverrides().at( wxT( "SHEETPATH" ) );
1333
1334 // We are feeding it one layer at the start here to silence a logic check
1335 PLOTTER* plotter = StartPlotBoard( brd, &plotOpts, layer, layerName, outPath, sheetName,
1336 sheetPath );
1337
1338 if( plotter )
1339 {
1340 PlotBoardLayers( brd, plotter, aGerberJob->m_plotLayerSequence, plotOpts );
1341 plotter->EndPlot();
1342 }
1343 else
1344 {
1345 m_reporter->Report( wxString::Format( _( "Failed to plot to '%s'.\n" ), outPath ),
1348 }
1349
1350 delete plotter;
1351
1352 return exitCode;
1353}
1354
1357
1358
1360{
1361 JOB_EXPORT_PCB_DRILL* aDrillJob = dynamic_cast<JOB_EXPORT_PCB_DRILL*>( aJob );
1362
1363 if( aDrillJob == nullptr )
1365
1366 BOARD* brd = getBoard( aDrillJob->m_filename );
1367
1368 if( !brd )
1370
1371 aJob->SetTitleBlock( brd->GetTitleBlock() );
1372
1373 wxString outPath = aDrillJob->GetFullOutputPath( brd->GetProject() );
1374
1375 if( !PATHS::EnsurePathExists( outPath ) )
1376 {
1377 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
1379 }
1380
1381 std::unique_ptr<GENDRILL_WRITER_BASE> drillWriter;
1382
1384 drillWriter = std::make_unique<EXCELLON_WRITER>( brd );
1385 else
1386 drillWriter = std::make_unique<GERBER_WRITER>( brd );
1387
1388 VECTOR2I offset;
1389
1391 offset = VECTOR2I( 0, 0 );
1392 else
1393 offset = brd->GetDesignSettings().GetAuxOrigin();
1394
1395 PLOT_FORMAT mapFormat = PLOT_FORMAT::PDF;
1396
1397 switch( aDrillJob->m_mapFormat )
1398 {
1399 case JOB_EXPORT_PCB_DRILL::MAP_FORMAT::POSTSCRIPT: mapFormat = PLOT_FORMAT::POST; break;
1400 case JOB_EXPORT_PCB_DRILL::MAP_FORMAT::GERBER_X2: mapFormat = PLOT_FORMAT::GERBER; break;
1401 case JOB_EXPORT_PCB_DRILL::MAP_FORMAT::DXF: mapFormat = PLOT_FORMAT::DXF; break;
1402 case JOB_EXPORT_PCB_DRILL::MAP_FORMAT::SVG: mapFormat = PLOT_FORMAT::SVG; break;
1403 default:
1404 case JOB_EXPORT_PCB_DRILL::MAP_FORMAT::PDF: mapFormat = PLOT_FORMAT::PDF; break;
1405 }
1406
1408 {
1410 switch( aDrillJob->m_zeroFormat )
1411 {
1414 break;
1417 break;
1420 break;
1422 default:
1424 break;
1425 }
1426
1427 DRILL_PRECISION precision;
1428
1430 precision = precisionListForInches;
1431 else
1432 precision = precisionListForMetric;
1433
1434 EXCELLON_WRITER* excellonWriter = dynamic_cast<EXCELLON_WRITER*>( drillWriter.get() );
1435
1436 if( excellonWriter == nullptr )
1438
1439 excellonWriter->SetFormat( aDrillJob->m_drillUnits == JOB_EXPORT_PCB_DRILL::DRILL_UNITS::MM,
1440 zeroFmt, precision.m_Lhs, precision.m_Rhs );
1441 excellonWriter->SetOptions( aDrillJob->m_excellonMirrorY,
1442 aDrillJob->m_excellonMinimalHeader,
1443 offset, aDrillJob->m_excellonCombinePTHNPTH );
1444 excellonWriter->SetRouteModeForOvalHoles( aDrillJob->m_excellonOvalDrillRoute );
1445 excellonWriter->SetMapFileFormat( mapFormat );
1446
1447 if( !excellonWriter->CreateDrillandMapFilesSet( outPath, true, aDrillJob->m_generateMap,
1448 m_reporter ) )
1449 {
1451 }
1452 }
1454 {
1455 GERBER_WRITER* gerberWriter = dynamic_cast<GERBER_WRITER*>( drillWriter.get() );
1456
1457 if( gerberWriter == nullptr )
1459
1460 // Set gerber precision: only 5 or 6 digits for mantissa are allowed
1461 // (SetFormat() accept 5 or 6, and any other value set the precision to 5)
1462 // the integer part precision is always 4, and units always mm
1463 gerberWriter->SetFormat( aDrillJob->m_gerberPrecision );
1464 gerberWriter->SetOptions( offset );
1465 gerberWriter->SetMapFileFormat( mapFormat );
1466
1467 if( !gerberWriter->CreateDrillandMapFilesSet( outPath, true, aDrillJob->m_generateMap,
1468 aDrillJob->m_generateTenting, m_reporter ) )
1469 {
1471 }
1472 }
1473
1474 return CLI::EXIT_CODES::OK;
1475}
1476
1477
1479{
1480 JOB_EXPORT_PCB_POS* aPosJob = dynamic_cast<JOB_EXPORT_PCB_POS*>( aJob );
1481
1482 if( aPosJob == nullptr )
1484
1485 BOARD* brd = getBoard( aPosJob->m_filename );
1486
1487 if( !brd )
1489
1490 aJob->SetTitleBlock( brd->GetTitleBlock() );
1491
1492 if( aPosJob->GetConfiguredOutputPath().IsEmpty() )
1493 {
1494 wxFileName fn = brd->GetFileName();
1495 fn.SetName( fn.GetName() );
1496
1499 else if( aPosJob->m_format == JOB_EXPORT_PCB_POS::FORMAT::CSV )
1500 fn.SetExt( FILEEXT::CsvFileExtension );
1501 else if( aPosJob->m_format == JOB_EXPORT_PCB_POS::FORMAT::GERBER )
1502 fn.SetExt( FILEEXT::GerberFileExtension );
1503
1504 aPosJob->SetWorkingOutputPath( fn.GetFullName() );
1505 }
1506
1507 wxString outPath = aPosJob->GetFullOutputPath( brd->GetProject() );
1508
1509 if( !PATHS::EnsurePathExists( outPath, true ) )
1510 {
1511 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
1513 }
1514
1517 {
1518 wxFileName fn( outPath );
1519 wxString baseName = fn.GetName();
1520
1521 auto exportPlaceFile =
1522 [&]( bool frontSide, bool backSide, const wxString& curr_outPath ) -> bool
1523 {
1524 FILE* file = wxFopen( curr_outPath, wxS( "wt" ) );
1525 wxCHECK( file, false );
1526
1527 PLACE_FILE_EXPORTER exporter( brd,
1529 aPosJob->m_smdOnly,
1531 aPosJob->m_excludeDNP,
1532 frontSide,
1533 backSide,
1536 aPosJob->m_negateBottomX );
1537
1538 std::string data = exporter.GenPositionData();
1539 fputs( data.c_str(), file );
1540 fclose( file );
1541
1542 return true;
1543 };
1544
1545 if( aPosJob->m_side == JOB_EXPORT_PCB_POS::SIDE::BOTH && !aPosJob->m_singleFile )
1546 {
1547 fn.SetName( PLACE_FILE_EXPORTER::DecorateFilename( baseName, true, false ) );
1548
1549 if( aPosJob->m_format == JOB_EXPORT_PCB_POS::FORMAT::CSV && !aPosJob->m_nakedFilename )
1550 fn.SetName( fn.GetName() + wxT( "-" ) + FILEEXT::FootprintPlaceFileExtension );
1551
1552 if( exportPlaceFile( true, false, fn.GetFullPath() ) )
1553 {
1554 m_reporter->Report( wxString::Format( _( "Wrote front position data to '%s'.\n" ),
1555 fn.GetFullPath() ),
1557
1558 aPosJob->AddOutput( fn.GetFullPath() );
1559 }
1560 else
1561 {
1563 }
1564
1565 fn.SetName( PLACE_FILE_EXPORTER::DecorateFilename( baseName, false, true ) );
1566
1567 if( aPosJob->m_format == JOB_EXPORT_PCB_POS::FORMAT::CSV && !aPosJob->m_nakedFilename )
1568 fn.SetName( fn.GetName() + wxT( "-" ) + FILEEXT::FootprintPlaceFileExtension );
1569
1570 if( exportPlaceFile( false, true, fn.GetFullPath() ) )
1571 {
1572 m_reporter->Report( wxString::Format( _( "Wrote back position data to '%s'.\n" ),
1573 fn.GetFullPath() ),
1575
1576 aPosJob->AddOutput( fn.GetFullPath() );
1577 }
1578 else
1579 {
1581 }
1582 }
1583 else
1584 {
1585 bool front = aPosJob->m_side == JOB_EXPORT_PCB_POS::SIDE::FRONT
1587
1588 bool back = aPosJob->m_side == JOB_EXPORT_PCB_POS::SIDE::BACK
1590
1591 if( !aPosJob->m_nakedFilename )
1592 {
1593 fn.SetName( PLACE_FILE_EXPORTER::DecorateFilename( fn.GetName(), front, back ) );
1594
1596 fn.SetName( fn.GetName() + wxT( "-" ) + FILEEXT::FootprintPlaceFileExtension );
1597 }
1598
1599 if( exportPlaceFile( front, back, fn.GetFullPath() ) )
1600 {
1601 m_reporter->Report( wxString::Format( _( "Wrote position data to '%s'.\n" ),
1602 fn.GetFullPath() ),
1604
1605 aPosJob->AddOutput( fn.GetFullPath() );
1606 }
1607 else
1608 {
1610 }
1611 }
1612 }
1613 else if( aPosJob->m_format == JOB_EXPORT_PCB_POS::FORMAT::GERBER )
1614 {
1615 PLACEFILE_GERBER_WRITER exporter( brd );
1616 PCB_LAYER_ID gbrLayer = F_Cu;
1617 wxString outPath_base = outPath;
1618
1620 || aPosJob->m_side == JOB_EXPORT_PCB_POS::SIDE::BOTH )
1621 {
1622 if( aPosJob->m_side == JOB_EXPORT_PCB_POS::SIDE::BOTH || !aPosJob->m_nakedFilename )
1623 outPath = exporter.GetPlaceFileName( outPath, gbrLayer );
1624
1625 if( exporter.CreatePlaceFile( outPath, gbrLayer, aPosJob->m_gerberBoardEdge ) >= 0 )
1626 {
1627 m_reporter->Report( wxString::Format( _( "Wrote front position data to '%s'.\n" ),
1628 outPath ),
1630
1631 aPosJob->AddOutput( outPath );
1632 }
1633 else
1634 {
1636 }
1637 }
1638
1640 || aPosJob->m_side == JOB_EXPORT_PCB_POS::SIDE::BOTH )
1641 {
1642 gbrLayer = B_Cu;
1643
1644 outPath = outPath_base;
1645
1646 if( aPosJob->m_side == JOB_EXPORT_PCB_POS::SIDE::BOTH || !aPosJob->m_nakedFilename )
1647 outPath = exporter.GetPlaceFileName( outPath, gbrLayer );
1648
1649 if( exporter.CreatePlaceFile( outPath, gbrLayer, aPosJob->m_gerberBoardEdge ) >= 0 )
1650 {
1651 m_reporter->Report( wxString::Format( _( "Wrote back position data to '%s'.\n" ),
1652 outPath ),
1654
1655 aPosJob->AddOutput( outPath );
1656 }
1657 else
1658 {
1660 }
1661 }
1662 }
1663
1664 return CLI::EXIT_CODES::OK;
1665}
1666
1667
1669{
1670 JOB_FP_UPGRADE* upgradeJob = dynamic_cast<JOB_FP_UPGRADE*>( aJob );
1671
1672 if( upgradeJob == nullptr )
1674
1676
1677 if( !upgradeJob->m_outputLibraryPath.IsEmpty() )
1678 {
1679 if( wxFile::Exists( upgradeJob->m_outputLibraryPath )
1680 || wxDir::Exists( upgradeJob->m_outputLibraryPath) )
1681 {
1682 m_reporter->Report( _( "Output path must not conflict with existing path\n" ),
1685 }
1686 }
1687 else if( fileType != PCB_IO_MGR::KICAD_SEXP )
1688 {
1689 m_reporter->Report( _( "Output path must be specified to convert legacy and non-KiCad libraries\n" ),
1691
1693 }
1694
1696 {
1697 if( !wxDir::Exists( upgradeJob->m_libraryPath ) )
1698 {
1699 m_reporter->Report( _( "Footprint library path does not exist or is not accessible\n" ),
1702 }
1703
1705 FP_CACHE fpLib( &pcb_io, upgradeJob->m_libraryPath );
1706
1707 try
1708 {
1709 fpLib.Load();
1710 }
1711 catch( ... )
1712 {
1713 m_reporter->Report( _( "Unable to load library\n" ), RPT_SEVERITY_ERROR );
1715 }
1716
1717 bool shouldSave = upgradeJob->m_force;
1718
1719 for( const auto& footprint : fpLib.GetFootprints() )
1720 {
1721 if( footprint.second->GetFootprint()->GetFileFormatVersionAtLoad()
1723 {
1724 shouldSave = true;
1725 }
1726 }
1727
1728 if( shouldSave )
1729 {
1730 try
1731 {
1732 if( !upgradeJob->m_outputLibraryPath.IsEmpty() )
1733 fpLib.SetPath( upgradeJob->m_outputLibraryPath );
1734
1735 fpLib.Save();
1736 }
1737 catch( ... )
1738 {
1739 m_reporter->Report( _( "Unable to save library\n" ), RPT_SEVERITY_ERROR );
1741 }
1742 }
1743 else
1744 {
1745 m_reporter->Report( _( "Footprint library was not updated\n" ), RPT_SEVERITY_ERROR );
1746 }
1747 }
1748 else
1749 {
1750 if( !PCB_IO_MGR::ConvertLibrary( nullptr, upgradeJob->m_libraryPath,
1751 upgradeJob->m_outputLibraryPath, nullptr /* REPORTER */ ) )
1752 {
1753 m_reporter->Report( ( "Unable to convert library\n" ), RPT_SEVERITY_ERROR );
1755 }
1756 }
1757
1758 return CLI::EXIT_CODES::OK;
1759}
1760
1761
1763{
1764 JOB_FP_EXPORT_SVG* svgJob = dynamic_cast<JOB_FP_EXPORT_SVG*>( aJob );
1765
1766 if( svgJob == nullptr )
1768
1770 FP_CACHE fpLib( &pcb_io, svgJob->m_libraryPath );
1771
1772 if( svgJob->m_argLayers )
1773 {
1774 if( !svgJob->m_argLayers.value().empty() )
1775 svgJob->m_plotLayerSequence = convertLayerArg( svgJob->m_argLayers.value(), nullptr );
1776 else
1778 }
1779
1780 try
1781 {
1782 fpLib.Load();
1783 }
1784 catch( ... )
1785 {
1786 m_reporter->Report( _( "Unable to load library\n" ), RPT_SEVERITY_ERROR );
1788 }
1789
1790 wxString outPath = svgJob->GetFullOutputPath( nullptr );
1791
1792 if( !PATHS::EnsurePathExists( outPath, true ) )
1793 {
1794 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
1796 }
1797
1798 int exitCode = CLI::EXIT_CODES::OK;
1799 bool singleFpPlotted = false;
1800
1801 for( const auto& [fpName, fpCacheEntry] : fpLib.GetFootprints() )
1802 {
1803 if( !svgJob->m_footprint.IsEmpty() )
1804 {
1805 // skip until we find the right footprint
1806 if( fpName != svgJob->m_footprint )
1807 continue;
1808 else
1809 singleFpPlotted = true;
1810 }
1811
1812 exitCode = doFpExportSvg( svgJob, fpCacheEntry->GetFootprint().get() );
1813
1814 if( exitCode != CLI::EXIT_CODES::OK )
1815 break;
1816 }
1817
1818 if( !svgJob->m_footprint.IsEmpty() && !singleFpPlotted )
1819 {
1820 m_reporter->Report( _( "The given footprint could not be found to export." ) + wxS( "\n" ),
1822 }
1823
1824 return CLI::EXIT_CODES::OK;
1825}
1826
1827
1829{
1830 // the hack for now is we create fake boards containing the footprint and plot the board
1831 // until we refactor better plot api later
1832 std::unique_ptr<BOARD> brd;
1833 brd.reset( CreateEmptyBoard() );
1834 brd->GetProject()->ApplyTextVars( aSvgJob->GetVarOverrides() );
1835 brd->SynchronizeProperties();
1836
1837 FOOTPRINT* fp = dynamic_cast<FOOTPRINT*>( aFootprint->Clone() );
1838
1839 if( fp == nullptr )
1841
1842 fp->SetLink( niluuid );
1843 fp->SetFlags( IS_NEW );
1844 fp->SetParent( brd.get() );
1845
1846 for( PAD* pad : fp->Pads() )
1847 {
1848 pad->SetLocalRatsnestVisible( false );
1849 pad->SetNetCode( 0 );
1850 }
1851
1852 fp->SetOrientation( ANGLE_0 );
1853 fp->SetPosition( VECTOR2I( 0, 0 ) );
1854
1855 brd->Add( fp, ADD_MODE::INSERT, true );
1856
1857 wxFileName outputFile;
1858 outputFile.SetPath( aSvgJob->GetFullOutputPath(nullptr) );
1859 outputFile.SetName( aFootprint->GetFPID().GetLibItemName().wx_str() );
1860 outputFile.SetExt( FILEEXT::SVGFileExtension );
1861
1862 m_reporter->Report( wxString::Format( _( "Plotting footprint '%s' to '%s'\n" ),
1863 aFootprint->GetFPID().GetLibItemName().wx_str(),
1864 outputFile.GetFullPath() ),
1866
1867 PCB_PLOT_PARAMS plotOpts;
1868 PCB_PLOTTER::PlotJobToPlotOpts( plotOpts, aSvgJob, *m_reporter );
1869
1870 // always fixed for the svg plot
1871 plotOpts.SetPlotFrameRef( false );
1872 plotOpts.SetSvgFitPageToBoard( true );
1873 plotOpts.SetMirror( false );
1874 plotOpts.SetSkipPlotNPTH_Pads( false );
1875
1876 if( plotOpts.GetSketchPadsOnFabLayers() )
1877 {
1878 plotOpts.SetPlotPadNumbers( true );
1879 }
1880
1881 PCB_PLOTTER plotter( brd.get(), m_reporter, plotOpts );
1882
1883 if( !plotter.Plot( outputFile.GetFullPath(),
1884 aSvgJob->m_plotLayerSequence,
1886 false,
1887 true,
1888 wxEmptyString, wxEmptyString,
1889 wxEmptyString ) )
1890 {
1891 m_reporter->Report( _( "Error creating svg file" ) + wxS( "\n" ), RPT_SEVERITY_ERROR );
1893 }
1894
1895 return CLI::EXIT_CODES::OK;
1896}
1897
1898
1900{
1901 JOB_PCB_DRC* drcJob = dynamic_cast<JOB_PCB_DRC*>( aJob );
1902
1903 if( drcJob == nullptr )
1905
1906 BOARD* brd = getBoard( drcJob->m_filename );
1907
1908 if( !brd )
1910
1911 aJob->SetTitleBlock( brd->GetTitleBlock() );
1912 brd->GetProject()->ApplyTextVars( aJob->GetVarOverrides() );
1913 brd->SynchronizeProperties();
1914
1915 if( drcJob->GetConfiguredOutputPath().IsEmpty() )
1916 {
1917 wxFileName fn = brd->GetFileName();
1918 fn.SetName( fn.GetName() + wxS( "-drc" ) );
1919
1921 fn.SetExt( FILEEXT::JsonFileExtension );
1922 else
1923 fn.SetExt( FILEEXT::ReportFileExtension );
1924
1925 drcJob->SetWorkingOutputPath( fn.GetFullName() );
1926 }
1927
1928 wxString outPath = drcJob->GetFullOutputPath( brd->GetProject() );
1929
1930 if( !PATHS::EnsurePathExists( outPath, true ) )
1931 {
1932 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
1934 }
1935
1936 EDA_UNITS units;
1937
1938 switch( drcJob->m_units )
1939 {
1940 case JOB_PCB_DRC::UNITS::INCH: units = EDA_UNITS::INCH; break;
1941 case JOB_PCB_DRC::UNITS::MILS: units = EDA_UNITS::MILS; break;
1942 case JOB_PCB_DRC::UNITS::MM: units = EDA_UNITS::MM; break;
1943 default: units = EDA_UNITS::MM; break;
1944 }
1945
1946 std::shared_ptr<DRC_ENGINE> drcEngine = brd->GetDesignSettings().m_DRCEngine;
1947 std::unique_ptr<NETLIST> netlist = std::make_unique<NETLIST>();
1948
1949 drcEngine->SetDrawingSheet( getDrawingSheetProxyView( brd ) );
1950
1951 // BOARD_COMMIT uses TOOL_MANAGER to grab the board internally so we must give it one
1952 TOOL_MANAGER* toolManager = new TOOL_MANAGER;
1953 toolManager->SetEnvironment( brd, nullptr, nullptr, Kiface().KifaceSettings(), nullptr );
1954
1955 BOARD_COMMIT commit( toolManager );
1956 bool checkParity = drcJob->m_parity;
1957 std::string netlist_str;
1958
1959 if( checkParity )
1960 {
1961 wxString annotateMsg = _( "Schematic parity tests require a fully annotated schematic." );
1962 netlist_str = annotateMsg;
1963
1964 // The KIFACE_NETLIST_SCHEMATIC function has some broken-ness that the schematic
1965 // frame's version does not, but it is the only one that works in CLI, so we use it
1966 // if we don't have the sch frame open.
1967 // TODO: clean this up, see https://gitlab.com/kicad/code/kicad/-/issues/19929
1968 if( m_kiway->Player( FRAME_SCH, false ) )
1969 {
1971 }
1972 else
1973 {
1974 wxFileName schematicPath( drcJob->m_filename );
1975 schematicPath.SetExt( FILEEXT::KiCadSchematicFileExtension );
1976
1977 if( !schematicPath.Exists() )
1978 schematicPath.SetExt( FILEEXT::LegacySchematicFileExtension );
1979
1980 if( !schematicPath.Exists() )
1981 {
1982 m_reporter->Report( _( "Failed to fetch schematic netlist for parity tests.\n" ),
1984 checkParity = false;
1985 }
1986 else
1987 {
1988 typedef bool ( *NETLIST_FN_PTR )( const wxString&, std::string& );
1990 NETLIST_FN_PTR netlister =
1991 (NETLIST_FN_PTR) eeschema->IfaceOrAddress( KIFACE_NETLIST_SCHEMATIC );
1992 ( *netlister )( schematicPath.GetFullPath(), netlist_str );
1993 }
1994 }
1995
1996 if( netlist_str == annotateMsg )
1997 {
1998 m_reporter->Report( wxString( netlist_str ) + wxT( "\n" ), RPT_SEVERITY_ERROR );
1999 checkParity = false;
2000 }
2001 }
2002
2003 if( checkParity )
2004 {
2005 try
2006 {
2007 STRING_LINE_READER* lineReader = new STRING_LINE_READER( netlist_str,
2008 _( "Eeschema netlist" ) );
2009 KICAD_NETLIST_READER netlistReader( lineReader, netlist.get() );
2010
2011 netlistReader.LoadNetlist();
2012 }
2013 catch( const IO_ERROR& )
2014 {
2015 m_reporter->Report( _( "Failed to fetch schematic netlist for parity tests.\n" ),
2017 checkParity = false;
2018 }
2019
2020 drcEngine->SetSchematicNetlist( netlist.get() );
2021 }
2022
2023 drcEngine->SetProgressReporter( nullptr );
2024 drcEngine->SetViolationHandler(
2025 [&]( const std::shared_ptr<DRC_ITEM>& aItem, VECTOR2I aPos, int aLayer,
2026 DRC_CUSTOM_MARKER_HANDLER* aCustomHandler )
2027 {
2028 PCB_MARKER* marker = new PCB_MARKER( aItem, aPos, aLayer );
2029 commit.Add( marker );
2030 } );
2031
2032 brd->RecordDRCExclusions();
2033 brd->DeleteMARKERs( true, true );
2034 drcEngine->RunTests( units, drcJob->m_reportAllTrackErrors, checkParity );
2035 drcEngine->ClearViolationHandler();
2036
2037 commit.Push( _( "DRC" ), SKIP_UNDO | SKIP_SET_DIRTY );
2038
2039 // Update the exclusion status on any excluded markers that still exist.
2040 brd->ResolveDRCExclusions( false );
2041
2042 std::shared_ptr<DRC_ITEMS_PROVIDER> markersProvider = std::make_shared<DRC_ITEMS_PROVIDER>(
2044
2045 std::shared_ptr<DRC_ITEMS_PROVIDER> ratsnestProvider =
2046 std::make_shared<DRC_ITEMS_PROVIDER>( brd, MARKER_BASE::MARKER_RATSNEST );
2047
2048 std::shared_ptr<DRC_ITEMS_PROVIDER> fpWarningsProvider =
2049 std::make_shared<DRC_ITEMS_PROVIDER>( brd, MARKER_BASE::MARKER_PARITY );
2050
2051 markersProvider->SetSeverities( drcJob->m_severity );
2052 ratsnestProvider->SetSeverities( drcJob->m_severity );
2053 fpWarningsProvider->SetSeverities( drcJob->m_severity );
2054
2055 m_reporter->Report( wxString::Format( _( "Found %d violations\n" ),
2056 markersProvider->GetCount() ),
2058 m_reporter->Report( wxString::Format( _( "Found %d unconnected items\n" ),
2059 ratsnestProvider->GetCount() ),
2061
2062 if( checkParity )
2063 {
2064 m_reporter->Report( wxString::Format( _( "Found %d schematic parity issues\n" ),
2065 fpWarningsProvider->GetCount() ),
2067 }
2068
2069 DRC_REPORT reportWriter( brd, units, markersProvider, ratsnestProvider, fpWarningsProvider );
2070
2071 bool wroteReport = false;
2072
2074 wroteReport = reportWriter.WriteJsonReport( outPath );
2075 else
2076 wroteReport = reportWriter.WriteTextReport( outPath );
2077
2078 if( !wroteReport )
2079 {
2080 m_reporter->Report( wxString::Format( _( "Unable to save DRC report to %s\n" ), outPath ),
2083 }
2084
2085 m_reporter->Report( wxString::Format( _( "Saved DRC Report to %s\n" ), outPath ),
2087
2088 if( drcJob->m_exitCodeViolations )
2089 {
2090 if( markersProvider->GetCount() > 0 || ratsnestProvider->GetCount() > 0
2091 || fpWarningsProvider->GetCount() > 0 )
2092 {
2094 }
2095 }
2096
2098}
2099
2100
2102{
2103 JOB_EXPORT_PCB_IPC2581* job = dynamic_cast<JOB_EXPORT_PCB_IPC2581*>( aJob );
2104
2105 if( job == nullptr )
2107
2108 BOARD* brd = getBoard( job->m_filename );
2109
2110 if( !brd )
2112
2113 aJob->SetTitleBlock( brd->GetTitleBlock() );
2114
2115 if( job->GetConfiguredOutputPath().IsEmpty() )
2116 {
2117 wxFileName fn = brd->GetFileName();
2118 fn.SetName( fn.GetName() );
2119 fn.SetExt( FILEEXT::Ipc2581FileExtension );
2120
2121 job->SetWorkingOutputPath( fn.GetName() );
2122 }
2123
2124 wxString outPath = job->GetFullOutputPath( brd->GetProject() );
2125
2126 if( !PATHS::EnsurePathExists( outPath, true ) )
2127 {
2128 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
2130 }
2131
2132 std::map<std::string, UTF8> props;
2133 props["units"] = job->m_units == JOB_EXPORT_PCB_IPC2581::IPC2581_UNITS::MM ? "mm" : "inch";
2134 props["sigfig"] = wxString::Format( "%d", job->m_precision );
2135 props["version"] = job->m_version == JOB_EXPORT_PCB_IPC2581::IPC2581_VERSION::C ? "C" : "B";
2136 props["OEMRef"] = job->m_colInternalId;
2137 props["mpn"] = job->m_colMfgPn;
2138 props["mfg"] = job->m_colMfg;
2139 props["dist"] = job->m_colDist;
2140 props["distpn"] = job->m_colDistPn;
2141
2142 wxString tempFile = wxFileName::CreateTempFileName( wxS( "pcbnew_ipc" ) );
2143 try
2144 {
2146 pi->SetProgressReporter( m_progressReporter );
2147 pi->SaveBoard( tempFile, brd, &props );
2148 }
2149 catch( const IO_ERROR& ioe )
2150 {
2151 m_reporter->Report( wxString::Format( _( "Error generating IPC-2581 file '%s'.\n%s" ),
2152 job->m_filename,
2153 ioe.What() ),
2155
2156 wxRemoveFile( tempFile );
2157
2159 }
2160
2161 if( job->m_compress )
2162 {
2163 wxFileName tempfn = outPath;
2164 tempfn.SetExt( FILEEXT::Ipc2581FileExtension );
2165 wxFileName zipfn = tempFile;
2166 zipfn.SetExt( "zip" );
2167
2168 {
2169 wxFFileOutputStream fnout( zipfn.GetFullPath() );
2170 wxZipOutputStream zip( fnout );
2171 wxFFileInputStream fnin( tempFile );
2172
2173 zip.PutNextEntry( tempfn.GetFullName() );
2174 fnin.Read( zip );
2175 }
2176
2177 wxRemoveFile( tempFile );
2178 tempFile = zipfn.GetFullPath();
2179 }
2180
2181 // If save succeeded, replace the original with what we just wrote
2182 if( !wxRenameFile( tempFile, outPath ) )
2183 {
2184 m_reporter->Report( wxString::Format( _( "Error generating IPC-2581 file '%s'.\n"
2185 "Failed to rename temporary file '%s." ),
2186 outPath,
2187 tempFile ),
2189 }
2190
2192}
2193
2194
2196{
2197 JOB_EXPORT_PCB_IPCD356* job = dynamic_cast<JOB_EXPORT_PCB_IPCD356*>( aJob );
2198
2199 if( job == nullptr )
2201
2202 BOARD* brd = getBoard( job->m_filename );
2203
2204 if( !brd )
2206
2207 aJob->SetTitleBlock( brd->GetTitleBlock() );
2208
2209 if( job->GetConfiguredOutputPath().IsEmpty() )
2210 {
2211 wxFileName fn = brd->GetFileName();
2212 fn.SetName( fn.GetName() );
2213 fn.SetExt( FILEEXT::IpcD356FileExtension );
2214
2215 job->SetWorkingOutputPath( fn.GetFullName() );
2216 }
2217
2218 wxString outPath = job->GetFullOutputPath( brd->GetProject() );
2219
2220 if( !PATHS::EnsurePathExists( outPath, true ) )
2221 {
2222 m_reporter->Report( _( "Failed to create output directory\n" ), RPT_SEVERITY_ERROR );
2224 }
2225
2226 IPC356D_WRITER exporter( brd );
2227
2228 bool success = exporter.Write( outPath );
2229
2230 if( success )
2231 {
2232 m_reporter->Report( _( "Successfully created IPC-D-356 file\n" ), RPT_SEVERITY_INFO );
2234 }
2235 else
2236 {
2237 m_reporter->Report( _( "Failed to create IPC-D-356 file\n" ), RPT_SEVERITY_ERROR );
2239 }
2240}
2241
2242
2244{
2245 JOB_EXPORT_PCB_ODB* job = dynamic_cast<JOB_EXPORT_PCB_ODB*>( aJob );
2246
2247 if( job == nullptr )
2249
2250 BOARD* brd = getBoard( job->m_filename );
2251
2252 if( !brd )
2254
2255 aJob->SetTitleBlock( brd->GetTitleBlock() );
2256
2257 wxString path = job->GetConfiguredOutputPath();
2258
2259 if( job->GetConfiguredOutputPath().IsEmpty() )
2260 {
2262 {
2263 // just basic folder name
2264 job->SetWorkingOutputPath( "odb" );
2265 }
2266 else
2267 {
2268 wxFileName fn( brd->GetFileName() );
2269 fn.SetName( fn.GetName() + wxS( "-odb" ) );
2270
2271 switch( job->m_compressionMode )
2272 {
2274 fn.SetExt( FILEEXT::ArchiveFileExtension );
2275 break;
2277 fn.SetExt( "tgz" );
2278 break;
2279 default:
2280 break;
2281 };
2282
2283 job->SetWorkingOutputPath( fn.GetFullName() );
2284 }
2285 }
2286
2288
2290}
2291
2292
2294{
2296 &aBrd->GetPageSettings(),
2297 aBrd->GetProject(),
2298 &aBrd->GetTitleBlock(),
2299 &aBrd->GetProperties() );
2300
2301 drawingSheet->SetSheetName( std::string() );
2302 drawingSheet->SetSheetPath( std::string() );
2303 drawingSheet->SetIsFirstPage( true );
2304
2305 drawingSheet->SetFileName( TO_UTF8( aBrd->GetFileName() ) );
2306
2307 return drawingSheet;
2308}
2309
2310
2311void PCBNEW_JOBS_HANDLER::loadOverrideDrawingSheet( BOARD* aBrd, const wxString& aSheetPath )
2312{
2313 // dont bother attempting to load a empty path, if there was one
2314 if( aSheetPath.IsEmpty() )
2315 return;
2316
2317 auto loadSheet =
2318 [&]( const wxString& path ) -> bool
2319 {
2322 resolver.SetProject( aBrd->GetProject() );
2324
2326 aBrd->GetProject()->GetProjectPath(),
2327 aBrd->GetEmbeddedFiles() );
2328 wxString msg;
2329
2330 if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, &msg ) )
2331 {
2332 m_reporter->Report( wxString::Format( _( "Error loading drawing sheet '%s'." ),
2333 path )
2334 + wxS( "\n" ) + msg + wxS( "\n" ),
2336 return false;
2337 }
2338
2339 return true;
2340 };
2341
2342 if( loadSheet( aSheetPath ) )
2343 return;
2344
2345 // failed loading custom path, revert back to default
2346 loadSheet( aBrd->GetProject()->GetProjectFile().m_BoardDrawingSheetFile );
2347}
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
KIFACE_BASE & Kiface()
Global KIFACE_BASE "get" accessor.
#define RANGE_SCALE_3D
This defines the range that all coord will have to be rendered.
Definition: board_adapter.h:66
PROJECTION_TYPE
Definition: camera.h:40
static wxString m_DrawingSheetFileName
the name of the drawing sheet file, or empty to use the default drawing sheet
Definition: base_screen.h:85
Helper class to handle information needed to display 3D board.
Definition: board_adapter.h:73
double BiuTo3dUnits() const noexcept
Board integer units To 3D units.
bool m_IsPreviewer
true if we're in a 3D preview panel, false for the standard 3D viewer
void SetBoard(BOARD *aBoard) noexcept
Set current board to be rendered.
EDA_3D_VIEWER_SETTINGS * m_Cfg
std::map< int, COLOR4D > m_ColorOverrides
allows to override color scheme colors
void Set3dCacheManager(S3D_CACHE *aCacheMgr) noexcept
Update the cache manager pointer.
Definition: board_adapter.h:84
virtual void Push(const wxString &aMessage=wxEmptyString, int aCommitFlags=0) override
Execute the changes.
std::shared_ptr< DRC_ENGINE > m_DRCEngine
const VECTOR2I & GetAuxOrigin() const
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
LSET GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
Definition: board.cpp:829
EMBEDDED_FILES * GetEmbeddedFiles() override
Definition: board.cpp:2584
const PAGE_INFO & GetPageSettings() const
Definition: board.h:715
void RecordDRCExclusions()
Scan existing markers and record data from any that are Excluded.
Definition: board.cpp:328
TITLE_BLOCK & GetTitleBlock()
Definition: board.h:721
BOX2I ComputeBoundingBox(bool aBoardEdgesOnly=false) const
Calculate the bounding box containing all board items (or board edge segments).
Definition: board.cpp:1763
const std::map< wxString, wxString > & GetProperties() const
Definition: board.h:369
const wxString & GetFileName() const
Definition: board.h:334
std::vector< PCB_MARKER * > ResolveDRCExclusions(bool aCreateMarkers)
Rebuild DRC markers from the serialized data in BOARD_DESIGN_SETTINGS.
Definition: board.cpp:345
const PCB_PLOT_PARAMS & GetPlotOptions() const
Definition: board.h:718
const wxString GetLayerName(PCB_LAYER_ID aLayer) const
Return the name of a aLayer.
Definition: board.cpp:614
PROJECT * GetProject() const
Definition: board.h:511
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
void SynchronizeProperties()
Copy the current project's text variables into the boards property cache.
Definition: board.cpp:2135
void DeleteMARKERs()
Delete all MARKERS from the board.
Definition: board.cpp:1439
constexpr const Vec GetCenter() const
Definition: box2.h:230
void SetProjection(PROJECTION_TYPE aProjection)
Definition: camera.h:206
void RotateY_T1(float aAngleInRadians)
Definition: camera.cpp:686
bool Zoom_T1(float aFactor)
Definition: camera.cpp:629
bool SetCurWindowSize(const wxSize &aSize)
Update the windows size of the camera.
Definition: camera.cpp:571
bool ViewCommand_T1(VIEW3D_TYPE aRequestedView)
Definition: camera.cpp:110
void RotateX_T1(float aAngleInRadians)
Definition: camera.cpp:680
void SetLookAtPos_T1(const SFVEC3F &aLookAtPos)
Definition: camera.h:162
const SFVEC3F & GetLookAtPos_T1() const
Definition: camera.h:167
void RotateZ_T1(float aAngleInRadians)
Definition: camera.cpp:692
bool ParametersChanged()
Definition: camera.cpp:730
COMMIT & Add(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Add a new item to the model.
Definition: commit.h:80
static void GenerateODBPPFiles(const JOB_EXPORT_PCB_ODB &aJob, BOARD *aBoard, PCB_EDIT_FRAME *aParentFrame=nullptr, PROGRESS_REPORTER *aProgressReporter=nullptr, REPORTER *aErrorReporter=nullptr)
The dialog to create footprint position files and choose options (one or 2 files, units and force all...
A dialog to set the plot options and create plot files in various formats.
Definition: dialog_plot.h:41
int ShowModal() override
bool WriteJsonReport(const wxString &aFullFileName)
Definition: drc_report.cpp:114
bool WriteTextReport(const wxString &aFullFileName)
Definition: drc_report.cpp:47
Helper to handle drill precision format in excellon files.
bool LoadDrawingSheet(const wxString &aFullFileName, wxString *aMsg, bool aAppend=false)
Populate the list with a custom layout or the default layout if no custom layout is available.
static DS_DATA_MODEL & GetTheInstance()
Return the instance of DS_DATA_MODEL used in the application.
void SetSheetPath(const std::string &aSheetPath)
Set the sheet path displayed in the title block.
void SetSheetName(const std::string &aSheetName)
Set the sheet name displayed in the title block.
void SetIsFirstPage(bool aIsFirstPage)
Change if this is first page.
void SetFileName(const std::string &aFileName)
Set the file name displayed in the title block.
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition: eda_item.h:135
virtual void SetParent(EDA_ITEM *aParent)
Definition: eda_item.h:111
Create Excellon drill, drill map, and drill report files.
void SetFormat(bool aMetric, ZEROS_FMT aZerosFmt=DECIMAL_FORMAT, int aLeftDigits=0, int aRightDigits=0)
Initialize internal parameters to match the given format.
bool CreateDrillandMapFilesSet(const wxString &aPlotDirectory, bool aGenDrill, bool aGenMap, REPORTER *aReporter=nullptr)
Create the full set of Excellon drill file for the board.
void SetOptions(bool aMirror, bool aMinimalHeader, const VECTOR2I &aOffset, bool aMerge_PTH_NPTH)
Initialize internal parameters to match drill options.
void SetRouteModeForOvalHoles(bool aUseRouteModeForOvalHoles)
wxString m_outputFile
Definition: exporter_step.h:53
Wrapper to expose an API for writing VRML files, without exposing all the many structures used in the...
Definition: export_vrml.h:33
bool ExportVRML_File(PROJECT *aProject, wxString *aMessages, const wxString &aFullFileName, double aMMtoWRMLunit, bool aIncludeUnspecified, bool aIncludeDNP, bool aExport3DFiles, bool aUseRelativePaths, const wxString &a3D_Subdir, double aXRef, double aYRef)
Exports the board and its footprint shapes 3D (vrml files only) as a vrml file.
Provide an extensible class to resolve 3D model paths.
void SetProgramBase(PGM_BASE *aBase)
Set a pointer to the application's PGM_BASE instance used to extract the local env vars.
wxString ResolvePath(const wxString &aFileName, const wxString &aWorkingPath, const EMBEDDED_FILES *aFiles)
Determine the full path of the given file name.
bool SetProject(const PROJECT *aProject, bool *flgChanged=nullptr)
Set the current KiCad project directory as the first entry in the model path list.
void SetPosition(const VECTOR2I &aPos) override
Definition: footprint.cpp:2456
void SetLink(const KIID &aLink)
Definition: footprint.h:845
void SetOrientation(const EDA_ANGLE &aNewAngle)
Definition: footprint.cpp:2538
EDA_ITEM * Clone() const override
Invoke a function on all children.
Definition: footprint.cpp:2178
std::deque< PAD * > & Pads()
Definition: footprint.h:211
const LIB_ID & GetFPID() const
Definition: footprint.h:253
void SetPath(const wxString &aPath)
void Save(FOOTPRINT *aFootprintFilter=nullptr)
Save the footprint cache or a single footprint from it to disk.
boost::ptr_map< wxString, FP_CACHE_ENTRY > & GetFootprints()
void UseIndividualShapes(bool aUnique)
Make pad shapes unique.
void UsePinNamesUnique(bool aUnique)
Make pin names unique.
void StoreOriginCoordsInFile(bool aStore)
Store coord origin in genCAD file.
void FlipBottomPads(bool aFlip)
Flip pad shapes on the bottom side.
void SetPlotOffet(VECTOR2I aOffset)
Set the coordinates offet when exporting items.
bool WriteFile(const wxString &aFullFileName)
Export a genCAD file.
void SetMapFileFormat(PLOT_FORMAT aMapFmt)
Initialize the format for the drill map file.
GERBER_JOBFILE_WRITER is a class used to create Gerber job file a Gerber job file stores info to make...
bool CreateJobFile(const wxString &aFullFilename)
Creates a Gerber job file.
void AddGbrFile(PCB_LAYER_ID aLayer, wxString &aFilename)
add a gerber file name and type in job file list
virtual bool EndPlot() override
Used to create Gerber drill files.
bool CreateDrillandMapFilesSet(const wxString &aPlotDirectory, bool aGenDrill, bool aGenMap, bool aGenTenting, REPORTER *aReporter=nullptr)
Create the full set of Excellon drill file for the board filenames are computed from the board name,...
void SetOptions(const VECTOR2I &aOffset)
Initialize internal parameters to match drill options.
void SetFormat(int aRightDigits=6)
Initialize internal parameters to match the given format.
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
Wrapper to expose an API for writing IPC-D356 files.
Definition: export_d356.h:54
bool Write(const wxString &aFilename)
Generates and writes the netlist to a given path.
void Register(const std::string &aJobTypeName, std::function< int(JOB *job)> aHandler, std::function< bool(JOB *job, wxWindow *aParent)> aConfigHandler)
PROGRESS_REPORTER * m_progressReporter
REPORTER * m_reporter
JOB_EXPORT_PCB_3D::FORMAT m_format
EXPORTER_STEP_PARAMS m_3dparams
Despite the name; also used for other formats.
ODB_COMPRESSION m_compressionMode
@ ALL_LAYERS_ONE_FILE
DEPRECATED MODE.
LSEQ m_plotOnAllLayersSequence
Used by SVG & PDF.
std::optional< wxString > m_argLayers
std::optional< wxString > m_argCommonLayers
LSEQ m_plotLayerSequence
Layers to include on all individual layer prints.
wxString m_libraryPath
wxString m_outputLibraryPath
bool m_parity
Definition: job_pcb_drc.h:33
bool m_reportAllTrackErrors
Definition: job_pcb_drc.h:32
VECTOR3D m_lightBottomIntensity
VECTOR3D m_lightTopIntensity
BG_STYLE m_bgStyle
VECTOR3D m_lightCameraIntensity
VECTOR3D m_rotation
bool m_proceduralTextures
wxString m_filename
VECTOR3D m_pivot
VECTOR3D m_lightSideIntensity
std::string m_colorPreset
bool m_exitCodeViolations
Definition: job_rc.h:53
int m_severity
Definition: job_rc.h:50
UNITS m_units
Definition: job_rc.h:49
OUTPUT_FORMAT m_format
Definition: job_rc.h:51
wxString m_filename
Definition: job_rc.h:48
An simple container class that lets us dispatch output jobs to kifaces.
Definition: job.h:182
void AddOutput(wxString aOutputPath)
Definition: job.h:209
wxString GetFullOutputPath(PROJECT *aProject) const
Returns the full output path for the job, taking into account the configured output path,...
Definition: job.cpp:100
wxString GetConfiguredOutputPath() const
Returns the configured output path for the job.
Definition: job.h:226
void SetTitleBlock(const TITLE_BLOCK &aTitleBlock)
Definition: job.h:197
void SetWorkingOutputPath(const wxString &aPath)
Sets a transient output path for the job, it takes priority over the configured output path when GetF...
Definition: job.h:232
const std::map< wxString, wxString > & GetVarOverrides() const
Definition: job.h:190
Read the new s-expression based KiCad netlist format.
virtual void LoadNetlist() override
Load the contents of the netlist file into aNetlist.
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
A minimalistic software bus for communications between various DLLs/DSOs (DSOs) within the same KiCad...
Definition: kiway.h:285
virtual KIWAY_PLAYER * Player(FRAME_T aFrameType, bool doCreate=true, wxTopLevelWindow *aParent=nullptr)
Return the KIWAY_PLAYER* given a FRAME_T.
Definition: kiway.cpp:406
virtual KIFACE * KiFACE(FACE_T aFaceId, bool doLoad=true)
Return the KIFACE* given a FACE_T.
Definition: kiway.cpp:201
@ FACE_SCH
eeschema DSO
Definition: kiway.h:292
virtual void ExpressMail(FRAME_T aDestination, MAIL_T aCommand, std::string &aPayload, wxWindow *aSource=nullptr)
Send aPayload to aDestination from aSource.
Definition: kiway.cpp:527
const UTF8 & GetLibItemName() const
Definition: lib_id.h:102
Instantiate the current locale within a scope in which you are expecting exceptions to be thrown.
Definition: locale_io.h:49
LSEQ is a sequence (and therefore also a set) of PCB_LAYER_IDs.
Definition: lseq.h:47
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
LSEQ UIOrder() const
Return the copper, technical and user layers in the order shown in layer widget.
Definition: lset.cpp:712
static LSET AllLayersMask()
Definition: lset.cpp:601
LSEQ SeqStackupForPlotting() const
Return the sequence that is typical for a bottom-to-top stack-up.
Definition: lset.cpp:388
static LSET AllNonCuMask()
Return a mask holding all layer minus CU layers.
Definition: lset.cpp:583
static LSET InternalCuMask()
Return a complete set of internal copper layers which is all Cu layers except F_Cu and B_Cu.
Definition: lset.cpp:561
static LSET AllCuMask(int aCuLayerCount=MAX_CU_LAYERS)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Definition: lset.cpp:572
static wxString Name(PCB_LAYER_ID aLayerId)
Return the fixed name association with aLayerId.
Definition: lset.cpp:188
@ MARKER_DRAWING_SHEET
Definition: marker_base.h:56
Definition: pad.h:54
static bool EnsurePathExists(const wxString &aPath, bool aPathToFile=false)
Attempts to create a given path if it does not exist.
Definition: paths.cpp:482
void populateGerberPlotOptionsFromJob(PCB_PLOT_PARAMS &aPlotOpts, JOB_EXPORT_PCB_GERBER *aJob)
int JobExportFpUpgrade(JOB *aJob)
DS_PROXY_VIEW_ITEM * getDrawingSheetProxyView(BOARD *aBrd)
void loadOverrideDrawingSheet(BOARD *brd, const wxString &aSheetPath)
PCBNEW_JOBS_HANDLER(KIWAY *aKiway)
BOARD * getBoard(const wxString &aPath=wxEmptyString)
LSEQ convertLayerArg(wxString &aLayerString, BOARD *aBoard) const
int JobExportRender(JOB *aJob)
int doFpExportSvg(JOB_FP_EXPORT_SVG *aSvgJob, const FOOTPRINT *aFootprint)
BOARD * GetBoard() const
The main frame for Pcbnew.
A #PLUGIN derivation for saving and loading Pcbnew s-expression formatted files.
static PCB_IO * PluginFind(PCB_FILE_T aFileType)
Return a #PLUGIN which the caller can use to import, export, save, or load design documents.
Definition: pcb_io_mgr.cpp:69
PCB_FILE_T
The set of file types that the PCB_IO_MGR knows about, and for which there has been a plugin written,...
Definition: pcb_io_mgr.h:56
@ KICAD_SEXP
S-expression Pcbnew file format.
Definition: pcb_io_mgr.h:58
static bool ConvertLibrary(std::map< std::string, UTF8 > *aOldFileProps, const wxString &aOldFilePath, const wxString &aNewFilePath, REPORTER *aReporter)
Convert a schematic symbol library to the latest KiCad format.
Definition: pcb_io_mgr.cpp:191
static PCB_FILE_T GuessPluginTypeFromLibPath(const wxString &aLibPath, int aCtl=0)
Return a plugin type given a footprint library's libPath.
Definition: pcb_io_mgr.cpp:136
bool Plot(const wxString &aOutputPath, const LSEQ &aLayersToPlot, const LSEQ &aCommonLayers, bool aUseGerberFileExtensions, bool aOutputPathIsSingle=false, std::optional< wxString > aLayerName=std::nullopt, std::optional< wxString > aSheetName=std::nullopt, std::optional< wxString > aSheetPath=std::nullopt)
Definition: pcb_plotter.cpp:49
static void PlotJobToPlotOpts(PCB_PLOT_PARAMS &aOpts, JOB_EXPORT_PCB_PLOT *aJob, REPORTER &aReporter)
Translate a JOB to PCB_PLOT_PARAMS.
Parameters and options when plotting/printing a board.
void SetDrillMarksType(DRILL_MARKS aVal)
LSEQ GetPlotOnAllLayersSequence() const
void SetSkipPlotNPTH_Pads(bool aSkip)
void SetLayerSelection(const LSET &aSelection)
void SetPlotReference(bool aFlag)
void SetUseGerberX2format(bool aUse)
void SetPlotOnAllLayersSequence(LSEQ aSeq)
void SetPlotFrameRef(bool aFlag)
void SetPlotPadNumbers(bool aFlag)
LSET GetLayerSelection() const
void SetDisableGerberMacros(bool aDisable)
void SetMirror(bool aFlag)
void SetGerberPrecision(int aPrecision)
void SetSubtractMaskFromSilk(bool aSubtract)
bool GetSketchPadsOnFabLayers() const
void SetPlotValue(bool aFlag)
void SetUseGerberProtelExtensions(bool aUse)
void SetIncludeGerberNetlistInfo(bool aUse)
void SetCreateGerberJobFile(bool aCreate)
bool m_PDFSingle
Generate a single PDF file for all layers.
void SetUseAuxOrigin(bool aAux)
void SetSvgFitPageToBoard(int aSvgFitPageToBoard)
bool GetUseGerberProtelExtensions() const
void SetFormat(PLOT_FORMAT aFormat)
virtual SETTINGS_MANAGER & GetSettingsManager() const
Definition: pgm_base.h:125
Used to create Gerber drill files.
const wxString GetPlaceFileName(const wxString &aFullBaseFilename, PCB_LAYER_ID aLayer) const
int CreatePlaceFile(const wxString &aFullFilename, PCB_LAYER_ID aLayer, bool aIncludeBrdEdges)
Create an pnp gerber file.
The ASCII format of the kicad place file is:
static wxString DecorateFilename(const wxString &aBaseName, bool aFront, bool aBack)
std::string GenPositionData()
build a string filled with the position data
Base plotter engine class.
Definition: plotter.h:105
virtual bool EndPlot()=0
wxString m_BoardDrawingSheetFile
PcbNew params.
Definition: project_file.h:170
static S3D_CACHE * Get3DCacheManager(PROJECT *aProject, bool updateProjDir=false)
Return a pointer to an instance of the 3D cache manager.
Definition: project_pcb.cpp:77
virtual const wxString GetProjectFullName() const
Return the full path and name of the project.
Definition: project.cpp:140
virtual const wxString GetProjectPath() const
Return the full path of the project.
Definition: project.cpp:146
virtual void ApplyTextVars(const std::map< wxString, wxString > &aVarsMap)
Applies the given var map, it will create or update existing vars.
Definition: project.cpp:101
virtual PROJECT_FILE & GetProjectFile() const
Definition: project.h:203
bool Redraw(bool aIsMoving, REPORTER *aStatusReporter, REPORTER *aWarningReporter) override
Redraw the view.
void SetCurWindowSize(const wxSize &aSize) override
Before each render, the canvas will tell the render what is the size of its windows,...
virtual REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED)
Report a string with a given severity.
Definition: reporter.h:102
T * GetAppSettings(const wxString &aFilename)
Return a handle to the a given settings by type.
PROJECT & Prj() const
A helper while we are not MDI-capable – return the one and only project.
Is a LINE_READER that reads from a multiline 8 bit wide std::string.
Definition: richio.h:253
Master controller class:
Definition: tool_manager.h:62
void SetEnvironment(EDA_ITEM *aModel, KIGFX::VIEW *aView, KIGFX::VIEW_CONTROLS *aViewControls, APP_SETTINGS_BASE *aSettings, TOOLS_HOLDER *aFrame)
Set the work environment (model, view, view controls and the parent window).
void Pan_T1(const SFVEC3F &aDeltaOffsetInc) override
Definition: track_ball.cpp:125
void SetT0_and_T1_current_T() override
This will set T0 and T1 with the current values.
Definition: track_ball.cpp:138
void Interpolate(float t) override
It will update the matrix to interpolate between T0 and T1 values.
Definition: track_ball.cpp:152
wxString wx_str() const
Definition: utf8.cpp:45
T y
Definition: vector3.h:64
T z
Definition: vector3.h:65
T x
Definition: vector3.h:63
wxString GetDefaultPlotExtension(PLOT_FORMAT aFormat)
Return the default plot extension for a format.
static DRILL_PRECISION precisionListForInches(2, 4)
static DRILL_PRECISION precisionListForMetric(3, 3)
std::function< void(PCB_MARKER *aMarker)> DRC_CUSTOM_MARKER_HANDLER
Definition: drc_engine.h:69
#define _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
#define IS_NEW
New item, just created.
EDA_UNITS
Definition: eda_units.h:46
static FILENAME_RESOLVER * resolver
Definition: export_idf.cpp:53
@ FRAME_PCB_EDITOR
Definition: frame_type.h:42
@ FRAME_SCH
Definition: frame_type.h:34
Classes used in drill files, map files and report files generation.
Classes used in drill files, map files and report files generation.
Classes used to generate a Gerber job file in JSON.
Classes used in place file generation.
static const std::string LegacySchematicFileExtension
static const std::string BrepFileExtension
static const std::string JpegFileExtension
static const std::string GerberJobFileExtension
static const std::string GerberFileExtension
static const std::string XaoFileExtension
static const std::string ReportFileExtension
static const std::string GltfBinaryFileExtension
static const std::string PngFileExtension
static const std::string FootprintPlaceFileExtension
static const std::string JsonFileExtension
static const std::string KiCadSchematicFileExtension
static const std::string CsvFileExtension
static const std::string Ipc2581FileExtension
static const std::string GencadFileExtension
static const std::string StlFileExtension
static const std::string IpcD356FileExtension
static const std::string PlyFileExtension
static const std::string StepFileExtension
static const std::string SVGFileExtension
static const std::string VrmlFileExtension
static const std::string ArchiveFileExtension
static const std::string KiCadPcbFileExtension
std::unique_ptr< T > IO_RELEASER
Helper to hold and release an IO_BASE object when exceptions are thrown.
Definition: io_mgr.h:33
@ KIFACE_NETLIST_SCHEMATIC
Definition: kiface_ids.h:56
KIID niluuid(0)
wxString LayerName(int aLayer)
Returns the default display name for a given layer.
Definition: layer_id.cpp:31
@ LAYER_3D_BACKGROUND_TOP
Definition: layer_ids.h:540
@ LAYER_3D_BACKGROUND_BOTTOM
Definition: layer_ids.h:539
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ F_CrtYd
Definition: layer_ids.h:116
@ B_Adhes
Definition: layer_ids.h:103
@ F_Paste
Definition: layer_ids.h:104
@ F_Adhes
Definition: layer_ids.h:102
@ B_Mask
Definition: layer_ids.h:98
@ B_Cu
Definition: layer_ids.h:65
@ F_Mask
Definition: layer_ids.h:97
@ B_Paste
Definition: layer_ids.h:105
@ F_Fab
Definition: layer_ids.h:119
@ F_SilkS
Definition: layer_ids.h:100
@ B_CrtYd
Definition: layer_ids.h:115
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
@ B_SilkS
Definition: layer_ids.h:101
@ F_Cu
Definition: layer_ids.h:64
@ B_Fab
Definition: layer_ids.h:118
This file contains miscellaneous commonly used macros and functions.
@ MAIL_SCH_GET_NETLIST
Definition: mail_type.h:49
static const int ERR_ARGS
Definition: exit_codes.h:31
static const int OK
Definition: exit_codes.h:30
static const int ERR_RC_VIOLATIONS
Rules check violation count was greater than 0.
Definition: exit_codes.h:37
static const int ERR_INVALID_INPUT_FILE
Definition: exit_codes.h:33
static const int SUCCESS
Definition: exit_codes.h:29
static const int ERR_INVALID_OUTPUT_CONFLICT
Definition: exit_codes.h:34
static const int ERR_UNKNOWN
Definition: exit_codes.h:32
#define SEXPR_BOARD_FILE_VERSION
Current s-expression file format version. 2 was the last legacy format version.
#define CTL_FOR_LIBRARY
Format output for a footprint library instead of clipboard or BOARD.
static DRILL_PRECISION precisionListForInches(2, 4)
static DRILL_PRECISION precisionListForMetric(3, 3)
SETTINGS_MANAGER * GetSettingsManager()
BOARD * CreateEmptyBoard()
Construct a default BOARD with a temporary (no filename) project.
BOARD * LoadBoard(const wxString &aFileName, bool aSetActive)
Loads a board from file This function identifies the file type by extension and determines the correc...
const wxString GetGerberProtelExtension(int aLayer)
Definition: pcbplot.cpp:43
void BuildPlotFileName(wxFileName *aFilename, const wxString &aOutputDir, const wxString &aSuffix, const wxString &aExtension)
Complete a plot filename.
Definition: pcbplot.cpp:379
PLOTTER * StartPlotBoard(BOARD *aBoard, const PCB_PLOT_PARAMS *aPlotOpts, int aLayer, const wxString &aLayerName, const wxString &aFullFileName, const wxString &aSheetName, const wxString &aSheetPath, const wxString &aPageName=wxT("1"), const wxString &aPageNumber=wxEmptyString, const int aPageCount=1)
Open a new plotfile using the options (and especially the format) specified in the options and prepar...
void PlotBoardLayers(BOARD *aBoard, PLOTTER *aPlotter, const LSEQ &aLayerSequence, const PCB_PLOT_PARAMS &aPlotOptions)
Plot a sequence of board layer IDs.
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
PLOT_FORMAT
The set of supported output plot formats.
Definition: plotter.h:65
Plotting engines similar to ps (PostScript, Gerber, svg)
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_INFO
@ RPT_SEVERITY_ACTION
#define SKIP_SET_DIRTY
Definition: sch_commit.h:43
#define SKIP_UNDO
Definition: sch_commit.h:41
const int scale
std::vector< FAB_LAYER_COLOR > dummy
MODEL3D_FORMAT_TYPE fileType(const char *aFileName)
#define TO_UTF8(wxstring)
Convert a wxString to a UTF8 encoded C string for all wxWidgets build modes.
Definition: string_utils.h:403
std::vector< KIGFX::COLOR4D > raytrace_lightColor
constexpr double IUTomm(int iu) const
Definition: base_units.h:86
constexpr int mmToIU(double mm) const
Definition: base_units.h:88
Implement a participant in the KIWAY alchemy.
Definition: kiway.h:152
Declaration for a track ball camera.
double DEG2RAD(double deg)
Definition: trigo.h:166
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695
Definition of file extensions used in Kicad.
glm::vec3 SFVEC3F
Definition: xv3d_types.h:44