KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
dialog_export_step.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) 2016 Cirilo Bernardo
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
25#include <wx/log.h>
26#include <wx/stdpaths.h>
27#include <wx/process.h>
28#include <wx/string.h>
29#include <wx/filedlg.h>
30
31#include <pgm_base.h>
32#include <board.h>
33#include <confirm.h>
34#include <kidialog.h>
36#include "dialog_export_step.h"
38#include <footprint.h>
39#include <kiface_base.h>
40#include <locale_io.h>
41#include <math/vector3.h>
42#include <pcb_edit_frame.h>
43#include <pcbnew_settings.h>
44#include <project/project_file.h> // LAST_PATH_TYPE
45#include <reporter.h>
46#include <trace_helpers.h>
49#include <filename_resolver.h>
50#include <core/map_helpers.h>
53
54
55// Maps m_choiceFormat selection to extension (and kicad-cli command)
56static const std::vector<wxString> c_formatCommand = { FILEEXT::StepFileExtension,
62
63// Maps file extensions to m_choiceFormat selection
64static const std::map<wxString, int> c_formatExtToChoice = { { FILEEXT::StepFileExtension, 0 },
71
72
73
74int DIALOG_EXPORT_STEP::m_toleranceLastChoice = -1; // Use default
75int DIALOG_EXPORT_STEP::m_formatLastChoice = -1; // Use default
90
91
92DIALOG_EXPORT_STEP::DIALOG_EXPORT_STEP( PCB_EDIT_FRAME* aEditFrame, const wxString& aBoardPath ) :
93 DIALOG_EXPORT_STEP( aEditFrame, aEditFrame, aBoardPath )
94{
95}
96
97
99 const wxString& aBoardPath,
100 JOB_EXPORT_PCB_3D* aJob ) :
101 DIALOG_EXPORT_STEP_BASE( aEditFrame ),
102 m_editFrame( aEditFrame ),
103 m_job( aJob ),
104 m_originUnits( 0 /* mm */ ),
105 m_boardPath( aBoardPath )
106{
107 if( !m_job )
108 {
109 m_browseButton->SetBitmap( KiBitmapBundle( BITMAPS::small_folder ) );
110 SetupStandardButtons( { { wxID_OK, _( "Export" ) },
111 { wxID_CANCEL, _( "Close" ) } } );
112
113
114 // Build default output file name
115 // (last saved filename in project or built from board filename)
117
118 if( path.IsEmpty() )
119 {
120 wxFileName brdFile( m_editFrame->GetBoard()->GetFileName() );
121 brdFile.SetExt( wxT( "step" ) );
122 path = brdFile.GetFullPath();
123 }
124
125 m_outputFileName->SetValue( path );
126 }
127 else
128 {
129 SetTitle( m_job->GetSettingsDialogTitle() );
130
131 m_browseButton->Hide();
133 }
134
135 // DIALOG_SHIM needs a unique hash_key because classname will be the same for both job and
136 // non-job versions (which have different sizes).
137 m_hash_key = TO_UTF8( GetTitle() );
138
139 Layout();
140 bSizerSTEPFile->Fit( this );
141
142 SetFocus();
143
144 if( !m_job )
145 {
147
149
150 switch( m_origin )
151 {
152 default:
153 case STEP_ORIGIN_PLOT_AXIS: m_rbDrillAndPlotOrigin->SetValue( true ); break;
154 case STEP_ORIGIN_GRID_AXIS: m_rbGridOrigin->SetValue( true ); break;
155 case STEP_ORIGIN_USER: m_rbUserDefinedOrigin->SetValue( true ); break;
156 case STEP_ORIGIN_BOARD_CENTER: m_rbBoardCenterOrigin->SetValue( true ); break;
157 }
158
164
165 m_txtNetFilter->SetValue( m_netFilter );
166 m_cbOptimizeStep->SetValue( m_optimizeStep );
169 m_cbExportTracks->SetValue( m_exportTracks );
170 m_cbExportPads->SetValue( m_exportPads );
171 m_cbExportZones->SetValue( m_exportZones );
175 m_cbFuseShapes->SetValue( m_fuseShapes );
177 m_cbFillAllVias->SetValue( m_fillAllVias );
179 m_cbRemoveDNP->SetValue( m_noDNP );
182
184
185 switch( m_componentMode )
186 {
187 case COMPONENT_MODE::EXPORT_ALL: m_rbAllComponents->SetValue( true ); break;
188 case COMPONENT_MODE::EXPORT_SELECTED: m_rbOnlySelected->SetValue( true ); break;
189 case COMPONENT_MODE::CUSTOM_FILTER: m_rbFilteredComponents->SetValue( true ); break;
190 }
191
192 // Sync the enabled states
193 wxCommandEvent dummy;
195
196 m_STEP_OrgUnitChoice->SetSelection( m_originUnits );
197 wxString tmpStr;
198 tmpStr << m_userOriginX;
199 m_STEP_Xorg->SetValue( tmpStr );
200 tmpStr = wxEmptyString;
201 tmpStr << m_userOriginY;
202 m_STEP_Yorg->SetValue( tmpStr );
203 }
204 else
205 {
207 m_rbDrillAndPlotOrigin->SetValue( true );
209 m_rbGridOrigin->SetValue( true );
210 else if( m_job->m_3dparams.m_Origin.x == 0.0 && m_job->m_3dparams.m_Origin.y == 0.0 )
211 m_rbBoardCenterOrigin->SetValue( true );
212 else
213 m_rbUserDefinedOrigin->SetValue( true );
214
217
220
238
241
242 wxCommandEvent dummy;
244
245 m_STEP_OrgUnitChoice->SetSelection( m_originUnits );
246
247 wxString tmpStr;
248 tmpStr << m_userOriginX;
249 m_STEP_Xorg->SetValue( tmpStr );
250 tmpStr = wxEmptyString;
251 tmpStr << m_userOriginY;
252 m_STEP_Yorg->SetValue( tmpStr );
253 }
254
255 wxString bad_scales;
256 size_t bad_count = 0;
257
258 for( FOOTPRINT* fp : m_editFrame->GetBoard()->Footprints() )
259 {
260 for( const FP_3DMODEL& model : fp->Models() )
261 {
262 if( model.m_Scale.x != 1.0 || model.m_Scale.y != 1.0 || model.m_Scale.z != 1.0 )
263 {
264 bad_scales.Append( wxS("\n") );
265 bad_scales.Append( model.m_Filename );
266 bad_count++;
267 }
268 }
269
270 if( bad_count >= 5 )
271 break;
272 }
273
274 if( !bad_scales.empty()
275 && !Pgm().GetCommonSettings()->m_DoNotShowAgain.scaled_3d_models_warning )
276 {
277 wxString extendedMsg = _( "Non-unity scaled models:" ) + wxT( "\n" ) + bad_scales;
278
279 KIDIALOG msgDlg( m_editFrame, _( "Scaled models detected. "
280 "Model scaling is not reliable for mechanical export." ),
281 _( "Model Scale Warning" ), wxOK | wxICON_WARNING );
282 msgDlg.SetExtendedMessage( extendedMsg );
283 msgDlg.DoNotShowCheckbox( __FILE__, __LINE__ );
284
285 msgDlg.ShowModal();
286
287 if( msgDlg.DoNotShowAgain() )
289 }
290
291 if( m_toleranceLastChoice >= 0 )
293
294 if( m_formatLastChoice >= 0 )
295 m_choiceFormat->SetSelection( m_formatLastChoice );
296 else
297 // ensure the selected fmt and the output file ext are synchronized the first time
298 // the dialog is opened
300
301 // Now all widgets have the size fixed, call FinishDialogSettings
303}
304
305
307{
308 GetOriginOption(); // Update m_origin member.
309
310 PCBNEW_SETTINGS* cfg = nullptr;
311
312 try
313 {
315 }
316 catch( const std::runtime_error& e )
317 {
318 wxFAIL_MSG( e.what() );
319 }
320
321 if( !m_job ) // dont save mru if its a job dialog
322 {
323 if( cfg )
324 {
325 cfg->m_ExportStep.origin_mode = static_cast<int>( m_origin );
326 cfg->m_ExportStep.origin_units = m_STEP_OrgUnitChoice->GetSelection();
329
330 double val = 0.0;
331
332 m_STEP_Xorg->GetValue().ToDouble( &val );
333 cfg->m_ExportStep.origin_x = val;
334
335 m_STEP_Yorg->GetValue().ToDouble( &val );
336 cfg->m_ExportStep.origin_y = val;
337
339 cfg->m_ExportStep.no_dnp = m_cbRemoveDNP->GetValue();
340 }
341
342 m_netFilter = m_txtNetFilter->GetValue();
344 m_formatLastChoice = m_choiceFormat->GetSelection();
345 m_optimizeStep = m_cbOptimizeStep->GetValue();
346 m_exportBoardBody = m_cbExportBody->GetValue();
348 m_exportTracks = m_cbExportTracks->GetValue();
349 m_exportPads = m_cbExportPads->GetValue();
350 m_exportZones = m_cbExportZones->GetValue();
354 m_fuseShapes = m_cbFuseShapes->GetValue();
356 m_fillAllVias = m_cbFillAllVias->GetValue();
358
359 if( m_rbAllComponents->GetValue() )
361 else if( m_rbOnlySelected->GetValue() )
363 else
365 }
366}
367
368
370{
372
373 if( m_rbDrillAndPlotOrigin->GetValue() )
375 else if( m_rbGridOrigin->GetValue() )
377 else if( m_rbUserDefinedOrigin->GetValue() )
379 else if( m_rbBoardCenterOrigin->GetValue() )
381
382 return m_origin;
383}
384
385
386void PCB_EDIT_FRAME::OnExportSTEP( wxCommandEvent& event )
387{
388 wxFileName brdFile = GetBoard()->GetFileName();
389
390 // The project filename (.kicad_pro) of the auto saved board filename, if it is created
391 wxFileName autosaveProjFile;
392
393 if( GetScreen()->IsContentModified() || brdFile.GetFullPath().empty() )
394 {
395 if( !doAutoSave() )
396 {
397 DisplayErrorMessage( this, _( "STEP export failed! "
398 "Please save the PCB and try again" ) );
399 return;
400 }
401
402 wxString autosaveFileName = FILEEXT::AutoSaveFilePrefix + brdFile.GetName();
403
404 // Create a dummy .kicad_pro file for this auto saved board file.
405 // this is useful to use some settings (like project path and name)
406 // Because doAutoSave() works, the target directory exists and is writable
407 autosaveProjFile = brdFile;
408 autosaveProjFile.SetName( autosaveFileName );
409 autosaveProjFile.SetExt( "kicad_pro" );
410
411 // Use auto-saved board for export
412 GetSettingsManager()->SaveProjectCopy( autosaveProjFile.GetFullPath(), GetBoard()->GetProject() );
413 brdFile.SetName( autosaveFileName );
414 }
415
416 DIALOG_EXPORT_STEP dlg( this, brdFile.GetFullPath() );
417 dlg.ShowModal();
418
419 // If a dummy .kicad_pro file is created, delete it now it is useless.
420 if( !autosaveProjFile.GetFullPath().IsEmpty() )
421 wxRemoveFile( autosaveProjFile.GetFullPath() );
422}
423
424
425void DIALOG_EXPORT_STEP::onUpdateUnits( wxUpdateUIEvent& aEvent )
426{
427 aEvent.Enable( m_rbUserDefinedOrigin->GetValue() );
428}
429
430
431void DIALOG_EXPORT_STEP::onUpdateXPos( wxUpdateUIEvent& aEvent )
432{
433 aEvent.Enable( m_rbUserDefinedOrigin->GetValue() );
434}
435
436
437void DIALOG_EXPORT_STEP::onUpdateYPos( wxUpdateUIEvent& aEvent )
438{
439 aEvent.Enable( m_rbUserDefinedOrigin->GetValue() );
440}
441
442
443void DIALOG_EXPORT_STEP::onBrowseClicked( wxCommandEvent& aEvent )
444{
445 // clang-format off
446 wxString filter = _( "STEP files" )
448 + _( "Binary glTF files" )
450 + _( "XAO files" )
452 + _( "BREP (OCCT) files" )
454 + _( "PLY files" )
456 + _( "STL files" )
458 // clang-format on
459
460 // Build the absolute path of current output directory to preselect it in the file browser.
461 wxString path = ExpandEnvVarSubstitutions( m_outputFileName->GetValue(), &Prj() );
462 wxFileName fn( Prj().AbsolutePath( path ) );
463
464 wxFileDialog dlg( this, _( "3D Model Output File" ), fn.GetPath(), fn.GetFullName(), filter,
465 wxFD_SAVE );
466
467 if( dlg.ShowModal() == wxID_CANCEL )
468 return;
469
470 path = dlg.GetPath();
471 m_outputFileName->SetValue( path );
472
473 fn = wxFileName( path );
474
475 if( auto formatChoice = get_opt( c_formatExtToChoice, fn.GetExt().Lower() ) )
476 m_choiceFormat->SetSelection( *formatChoice );
477}
478
479
480void DIALOG_EXPORT_STEP::onFormatChoice( wxCommandEvent& event )
481{
483}
484
485
487{
488 wxString newExt = c_formatCommand[m_choiceFormat->GetSelection()];
489 wxString path = m_outputFileName->GetValue();
490
491 int sepIdx = std::max( path.Find( '/', true ), path.Find( '\\', true ) );
492 int dotIdx = path.Find( '.', true );
493
494 if( dotIdx == -1 || dotIdx < sepIdx )
495 path << '.' << newExt;
496 else
497 path = path.Mid( 0, dotIdx ) << '.' << newExt;
498
499 m_outputFileName->SetValue( path );
501}
502
503
504void DIALOG_EXPORT_STEP::onCbExportComponents( wxCommandEvent& event )
505{
506 bool enable = m_cbExportComponents->GetValue();
507
508 m_rbAllComponents->Enable( enable );
509 m_rbOnlySelected->Enable( enable );
510 m_rbFilteredComponents->Enable( enable );
511 m_txtComponentFilter->Enable( enable && m_rbFilteredComponents->GetValue() );
512}
513
514
515void DIALOG_EXPORT_STEP::OnComponentModeChange( wxCommandEvent& event )
516{
517 m_txtComponentFilter->Enable( m_rbFilteredComponents->GetValue() );
518}
519
520
521void DIALOG_EXPORT_STEP::onExportButton( wxCommandEvent& aEvent )
522{
523 if( !m_job )
524 {
525 wxString path = m_outputFileName->GetValue();
527
528 // Build the absolute path of current output directory to preselect it in the file browser.
529 std::function<bool( wxString* )> textResolver =
530 [&]( wxString* token ) -> bool
531 {
532 return m_editFrame->GetBoard()->ResolveTextVar( token, 0 );
533 };
534
535 path = ExpandTextVars( path, &textResolver );
537 path = Prj().AbsolutePath( path );
538
539 if( path.IsEmpty() )
540 {
541 DisplayErrorMessage( this, _( "No filename for output file" ) );
542 return;
543 }
544
545 m_netFilter = m_txtNetFilter->GetValue();
547
548 if( m_rbAllComponents->GetValue() )
550 else if( m_rbOnlySelected->GetValue() )
552 else
554
555 double tolerance; // default value in mm
557 m_formatLastChoice = m_choiceFormat->GetSelection();
558 m_optimizeStep = m_cbOptimizeStep->GetValue();
559 m_exportBoardBody = m_cbExportBody->GetValue();
561 m_exportTracks = m_cbExportTracks->GetValue();
562 m_exportPads = m_cbExportPads->GetValue();
563 m_exportZones = m_cbExportZones->GetValue();
567 m_fuseShapes = m_cbFuseShapes->GetValue();
569 m_fillAllVias = m_cbFillAllVias->GetValue();
570
571 switch( m_choiceTolerance->GetSelection() )
572 {
573 case 0: tolerance = 0.001; break;
574 default:
575 case 1: tolerance = 0.01; break;
576 case 2: tolerance = 0.1; break;
577 }
578
579 SHAPE_POLY_SET outline;
580 wxString msg;
581
582 // Check if the board outline is continuous
583 // max dist from one endPt to next startPt to build a closed shape:
584 int chainingEpsilon = pcbIUScale.mmToIU( tolerance );
585
586 // Arc to segment approx error (not critical here: we do not use the outline shape):
587 int maxError = pcbIUScale.mmToIU( 0.005 );
588 bool success = BuildBoardPolygonOutlines( m_editFrame->GetBoard(), outline, maxError,
589 chainingEpsilon, nullptr );
590 if( !success )
591 {
592 DisplayErrorMessage( this, wxString::Format(
593 _( "Board outline is missing or not closed using %.3f mm tolerance.\n"
594 "Run DRC for a full analysis." ), tolerance ) );
595 return;
596 }
597
598 wxFileName fn( Prj().AbsolutePath( path ) );
599
600 if( fn.FileExists() && !GetOverwriteFile() )
601 {
602 msg.Printf( _( "File '%s' already exists. Do you want overwrite this file?" ),
603 fn.GetFullPath() );
604
605 if( wxMessageBox( msg, _( "STEP/GLTF Export" ), wxYES_NO | wxICON_QUESTION, this ) == wxNO )
606 return;
607 }
608
609 wxFileName appK2S( wxStandardPaths::Get().GetExecutablePath() );
610 #ifdef __WXMAC__
611 // On macOS, we have standalone applications inside the main bundle, so we handle that here:
612 if( appK2S.GetPath().Find( "/Contents/Applications/pcbnew.app/Contents/MacOS" ) != wxNOT_FOUND )
613 {
614 appK2S.AppendDir( wxT( ".." ) );
615 appK2S.AppendDir( wxT( ".." ) );
616 appK2S.AppendDir( wxT( ".." ) );
617 appK2S.AppendDir( wxT( ".." ) );
618 appK2S.AppendDir( wxT( "MacOS" ) );
619 }
620 #else
621 if( wxGetEnv( wxT( "KICAD_RUN_FROM_BUILD_DIR" ), nullptr ) )
622 {
623 appK2S.RemoveLastDir();
624 appK2S.AppendDir( "kicad" );
625 }
626 #endif
627
628 appK2S.SetName( wxT( "kicad-cli" ) );
629 appK2S.Normalize( FN_NORMALIZE_FLAGS );
630
631 wxString cmdK2S = wxT( "\"" );
632 cmdK2S.Append( appK2S.GetFullPath() );
633 cmdK2S.Append( wxT( "\"" ) );
634
635 cmdK2S.Append( wxT( " pcb" ) );
636 cmdK2S.Append( wxT( " export" ) );
637
638 cmdK2S.Append( wxT( " " ) );
639 cmdK2S.Append( c_formatCommand[m_choiceFormat->GetSelection()] );
640
642 cmdK2S.Append( wxT( " --no-unspecified" ) );
643
644 if( GetNoDNPOption() )
645 cmdK2S.Append( wxT( " --no-dnp" ) );
646
647 if( GetSubstOption() )
648 cmdK2S.Append( wxT( " --subst-models" ) );
649
650 if( !m_optimizeStep )
651 cmdK2S.Append( wxT( " --no-optimize-step" ) );
652
653 if( !m_exportBoardBody )
654 cmdK2S.Append( wxT( " --no-board-body" ) );
655
656 if( !m_exportComponents )
657 cmdK2S.Append( wxT( " --no-components" ) );
658
659 if( m_exportTracks )
660 cmdK2S.Append( wxT( " --include-tracks" ) );
661
662 if( m_exportPads )
663 cmdK2S.Append( wxT( " --include-pads" ) );
664
665 if( m_exportZones )
666 cmdK2S.Append( wxT( " --include-zones" ) );
667
669 cmdK2S.Append( wxT( " --include-inner-copper" ) );
670
672 cmdK2S.Append( wxT( " --include-silkscreen" ) );
673
675 cmdK2S.Append( wxT( " --include-soldermask" ) );
676
677 if( m_fuseShapes )
678 cmdK2S.Append( wxT( " --fuse-shapes" ) );
679
680 if( m_cutViasInBody )
681 cmdK2S.Append( wxT( " --cut-vias-in-body" ) );
682
683 if( m_fillAllVias )
684 cmdK2S.Append( wxT( " --fill-all-vias" ) );
685
686 // Note: for some reason, using \" to insert a quote in a format string, under MacOS
687 // wxString::Format does not work. So use a %c format in string
688 int quote = '\'';
689 int dblquote = '"';
690
691 if( !m_netFilter.empty() )
692 {
693 cmdK2S.Append( wxString::Format( wxT( " --net-filter %c%s%c" ), dblquote, m_netFilter,
694 dblquote ) );
695 }
696
697 switch( m_componentMode )
698 {
700 {
701 wxArrayString components;
703
704 std::for_each( selection.begin(), selection.end(),
705 [&components]( EDA_ITEM* item )
706 {
707 if( item->Type() == PCB_FOOTPRINT_T )
708 components.push_back( static_cast<FOOTPRINT*>( item )->GetReference() );
709 } );
710
711 cmdK2S.Append( wxString::Format( wxT( " --component-filter %c%s%c" ), dblquote,
712 wxJoin( components, ',' ), dblquote ) );
713 break;
714 }
715
717 cmdK2S.Append( wxString::Format( wxT( " --component-filter %c%s%c" ), dblquote,
718 m_componentFilter, dblquote ) );
719 break;
720
721 default:
722 break;
723 }
724
725 switch( GetOriginOption() )
726 {
727 case STEP_ORIGIN_0:
728 wxFAIL_MSG( wxT( "Unsupported origin option: how did we get here?" ) );
729 break;
730
732 cmdK2S.Append( wxT( " --drill-origin" ) );
733 break;
734
736 cmdK2S.Append( wxT( " --grid-origin" ) );
737 break;
738
739 case STEP_ORIGIN_USER:
740 {
741 double xOrg = GetXOrg();
742 double yOrg = GetYOrg();
743
744 if( GetOrgUnitsChoice() == 1 )
745 {
746 // selected reference unit is in inches, and STEP units are mm
747 xOrg *= 25.4;
748 yOrg *= 25.4;
749 }
750
752 cmdK2S.Append( wxString::Format( wxT( " --user-origin=%c%.6fx%.6fmm%c" ),
753 quote, xOrg, yOrg, quote ) );
754 break;
755 }
756
758 {
759 BOX2I bbox = m_editFrame->GetBoard()->ComputeBoundingBox( true );
760 double xOrg = pcbIUScale.IUTomm( bbox.GetCenter().x );
761 double yOrg = pcbIUScale.IUTomm( bbox.GetCenter().y );
763
764 cmdK2S.Append( wxString::Format( wxT( " --user-origin=%c%.6fx%.6fmm%c" ),
765 quote, xOrg, yOrg, quote ) );
766 break;
767 }
768 }
769
770 {
772 cmdK2S.Append( wxString::Format( wxT( " --min-distance=%c%.3fmm%c" ),
773 quote, tolerance, quote ) );
774 }
775
776 // Output file path.
777 cmdK2S.Append( wxString::Format( wxT( " -f -o %c%s%c" ),
778 dblquote, fn.GetFullPath(), dblquote ) );
779
780
781 // Input file path.
782 cmdK2S.Append( wxString::Format( wxT( " %c%s%c" ), dblquote, m_boardPath, dblquote ) );
783
784 wxLogTrace( traceKiCad2Step, wxT( "export step command: %s" ), cmdK2S );
785
786 DIALOG_EXPORT_STEP_LOG* log = new DIALOG_EXPORT_STEP_LOG( this, cmdK2S );
787 log->ShowModal();
788 }
789 else
790 {
810
812 static_cast<EXPORTER_STEP_PARAMS::FORMAT>( m_choiceFormat->GetSelection() ) );
813
814 // ensure the main format on the job is populated
815 switch( m_job->m_3dparams.m_Format )
816 {
819 break;
822 break;
825 break;
828 break;
831 break;
834 break;
835 }
836
837 switch( GetOriginOption() )
838 {
839 case STEP_ORIGIN_0:
840 break;
843 break;
846 break;
847 case STEP_ORIGIN_USER:
848 {
849 double xOrg = GetXOrg();
850 double yOrg = GetYOrg();
851
852 if( GetOrgUnitsChoice() == 1 )
853 {
854 // selected reference unit is in inches, and STEP units are mm
855 xOrg *= 25.4;
856 yOrg *= 25.4;
857 }
858
859 m_job->m_3dparams.m_Origin = VECTOR2D( xOrg, yOrg );
860 break;
861 }
862
864 {
865 BOX2I bbox = m_editFrame->GetBoard()->ComputeBoundingBox( true );
866 double xOrg = pcbIUScale.IUTomm( bbox.GetCenter().x );
867 double yOrg = pcbIUScale.IUTomm( bbox.GetCenter().y );
869
870 m_job->m_3dparams.m_Origin = VECTOR2D( xOrg, yOrg );
871 break;
872 }
873 }
874
875 EndModal( wxID_OK );
876 }
877}
878
879
881{
883}
884
885
887{
889}
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition: bitmap.cpp:110
bool ResolveTextVar(wxString *token, int aDepth) const
Definition: board.cpp:434
BOX2I ComputeBoundingBox(bool aBoardEdgesOnly=false) const
Calculate the bounding box containing all board items (or board edge segments).
Definition: board.cpp:1763
const FOOTPRINTS & Footprints() const
Definition: board.h:338
const wxString & GetFileName() const
Definition: board.h:334
constexpr const Vec GetCenter() const
Definition: box2.h:230
DO_NOT_SHOW_AGAIN m_DoNotShowAgain
Class DIALOG_EXPORT_STEP_BASE.
wxRadioButton * m_rbFilteredComponents
wxRadioButton * m_rbDrillAndPlotOrigin
STD_BITMAP_BUTTON * m_browseButton
void onFormatChoice(wxCommandEvent &event) override
STEP_ORIGIN_OPTION GetOriginOption()
static bool m_exportSoldermask
int GetOrgUnitsChoice() const
void onCbExportComponents(wxCommandEvent &event) override
PCB_EDIT_FRAME * m_editFrame
void OnComponentModeChange(wxCommandEvent &event) override
DIALOG_EXPORT_STEP(PCB_EDIT_FRAME *aEditFrame, const wxString &aBoardPath)
static COMPONENT_MODE m_componentMode
void onUpdateXPos(wxUpdateUIEvent &aEvent) override
static bool m_exportComponents
STEP_ORIGIN_OPTION m_origin
JOB_EXPORT_PCB_3D * m_job
void onExportButton(wxCommandEvent &aEvent) override
static bool m_exportBoardBody
static bool m_exportSilkscreen
static wxString m_componentFilter
static int m_toleranceLastChoice
void onUpdateUnits(wxUpdateUIEvent &aEvent) override
static bool m_exportInnerCopper
void onUpdateYPos(wxUpdateUIEvent &aEvent) override
void onBrowseClicked(wxCommandEvent &aEvent) override
void SetupStandardButtons(std::map< int, wxString > aLabels={})
std::string m_hash_key
Definition: dialog_shim.h:194
void finishDialogSettings()
In all dialogs, we must call the same functions to fix minimal dlg size, the default position and per...
int ShowModal() override
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
JOB_EXPORT_PCB_3D::FORMAT m_format
wxString GetSettingsDialogTitle() const override
void SetStepFormat(EXPORTER_STEP_PARAMS::FORMAT aFormat)
EXPORTER_STEP_PARAMS m_3dparams
Despite the name; also used for other formats.
void SetConfiguredOutputPath(const wxString &aPath)
Sets the configured output path for the job, this path is always saved to file.
Definition: job.cpp:153
wxString GetConfiguredOutputPath() const
Returns the configured output path for the job.
Definition: job.h:226
Helper class to create more flexible dialogs, including 'do not show again' checkbox handling.
Definition: kidialog.h:43
bool DoNotShowAgain() const
Checks the 'do not show again' setting for the dialog.
Definition: kidialog.cpp:59
void DoNotShowCheckbox(wxString file, int line)
Shows the 'do not show again' checkbox.
Definition: kidialog.cpp:51
int ShowModal() override
Definition: kidialog.cpp:95
PROJECT & Prj() const
Return a reference to the PROJECT associated with this KIWAY.
Instantiate the current locale within a scope in which you are expecting exceptions to be thrown.
Definition: locale_io.h:49
DIALOG_EXPORT_STEP m_ExportStep
PCBNEW_SETTINGS * GetPcbNewSettings() const
PCB_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
BOARD * GetBoard() const
The main frame for Pcbnew.
void SetLastPath(LAST_PATH_TYPE aType, const wxString &aLastPath)
Set the path of the last file successfully read.
bool doAutoSave() override
Perform auto save when the board has been modified and not saved within the auto save interval.
void OnExportSTEP(wxCommandEvent &event)
Export the current BOARD to a STEP assembly.
wxString GetLastPath(LAST_PATH_TYPE aType)
Get the last path for a particular type.
bool IsContentModified() const override
Get if the current board has been modified but not saved.
SELECTION & GetCurrentSelection() override
Get the current selection from the canvas area.
virtual COMMON_SETTINGS * GetCommonSettings() const
Definition: pgm_base.cpp:687
virtual const wxString AbsolutePath(const wxString &aFileName) const
Fix up aFileName if it is relative to the project's directory to be an absolute path and filename.
Definition: project.cpp:370
ITER end()
Definition: selection.h:75
ITER begin()
Definition: selection.h:74
void SaveProjectCopy(const wxString &aFullPath, PROJECT *aProject=nullptr)
Save a copy of the current project under the given path.
Represent a set of closed polygons.
void SetBitmap(const wxBitmapBundle &aBmp)
void SetValue(const wxString &aValue) override
Set a new value in evaluator buffer and display it in the wxTextCtrl.
const wxString ExpandEnvVarSubstitutions(const wxString &aString, const PROJECT *aProject)
Replace any environment variable & text variable references with their values.
Definition: common.cpp:351
wxString ExpandTextVars(const wxString &aSource, const PROJECT *aProject, int aFlags)
Definition: common.cpp:59
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition: confirm.cpp:194
This file is part of the common library.
bool BuildBoardPolygonOutlines(BOARD *aBoard, SHAPE_POLY_SET &aOutlines, int aErrorMax, int aChainingEpsilon, OUTLINE_ERROR_HANDLER *aErrorHandler, bool aAllowUseArcsInPolygons)
Extract the board outlines and build a closed polygon from lines, arcs and circle items on edge cut l...
static const std::map< wxString, int > c_formatExtToChoice
static const std::vector< wxString > c_formatCommand
#define _(s)
static const std::string BrepFileExtension
static const std::string StepFileAbrvExtension
static const std::string XaoFileExtension
static const std::string GltfBinaryFileExtension
static const std::string StlFileExtension
static const std::string AutoSaveFilePrefix
static const std::string PlyFileExtension
static const std::string StepFileExtension
const wxChar *const traceKiCad2Step
Flag to enable KiCad2Step debug tracing.
This file is part of the common library.
std::optional< V > get_opt(const std::map< wxString, V > &aMap, const wxString &aKey)
Definition: map_helpers.h:34
KICOMMON_API double DoubleValueFromString(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnits, const wxString &aTextValue, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Convert aTextValue to a double.
Definition: eda_units.cpp:497
SETTINGS_MANAGER * GetSettingsManager()
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
@ LAST_PATH_STEP
Definition: project_file.h:52
std::vector< FAB_LAYER_COLOR > dummy
#define TO_UTF8(wxstring)
Convert a wxString to a UTF8 encoded C string for all wxWidgets build modes.
Definition: string_utils.h:403
constexpr double IUTomm(int iu) const
Definition: base_units.h:86
constexpr int mmToIU(double mm) const
Definition: base_units.h:88
wxLogTrace helper definitions.
VECTOR2< double > VECTOR2D
Definition: vector2d.h:694
wxString AddFileExtListToFilter(const std::vector< std::string > &aExts)
Build the wildcard extension file dialog wildcard filter to add to the base message dialog.
Definition of file extensions used in Kicad.
#define FN_NORMALIZE_FLAGS
Default flags to pass to wxFileName::Normalize().
Definition: wx_filename.h:39