/* * This program source code file is part of kicad2mcad * * Copyright (C) 2016 Cirilo Bernardo * Copyright (C) 2016-2021 KiCad Developers, see AUTHORS.txt for contributors. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, you may find one here: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html * or you may search the http://www.gnu.org website for the version 2 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include #include #include #include #include #include #include "kicad2step.h" #include "pcb/kicadpcb.h" #include "kicad2step_frame_base.h" #include "panel_kicad2step.h" #include // OpenCascade messenger #include // OpenCascade output messenger #include // In open cascade #include #define OCC_VERSION_MIN 0x070500 #if OCC_VERSION_HEX < OCC_VERSION_MIN #include #endif class KICAD2STEP_FRAME : public KICAD2STEP_FRAME_BASE { public: KICAD2STEP_FRAME( const wxString& title ); protected: virtual void OnOKButtonClick( wxCommandEvent& aEvent ) override; }; // Horrible hack until we decouple things more static PANEL_KICAD2STEP* openPanel = nullptr; void ReportMessage( const wxString& aMessage ) { if( openPanel != nullptr ) openPanel->AppendMessage( aMessage ); } class KiCadPrinter : public Message_Printer { protected: #if OCC_VERSION_HEX < OCC_VERSION_MIN virtual void Send( const TCollection_ExtendedString& theString, const Message_Gravity theGravity, const Standard_Boolean theToPutEol ) const override { Send (TCollection_AsciiString (theString), theGravity, theToPutEol); } virtual void Send( const TCollection_AsciiString& theString, const Message_Gravity theGravity, const Standard_Boolean theToPutEol) const override #else virtual void send( const TCollection_AsciiString& theString, const Message_Gravity theGravity ) const override #endif { if( theGravity >= Message_Info ) { ReportMessage( theString.ToCString() ); #if OCC_VERSION_HEX < OCC_VERSION_MIN if( theToPutEol ) ReportMessage( wxT( "\n" ) ); #else ReportMessage( wxT( "\n" ) ); #endif } if( theGravity >= Message_Alarm ) openPanel->m_error = true; if( theGravity == Message_Fail ) openPanel->m_fail = true; } }; KICAD2MCAD_PRMS::KICAD2MCAD_PRMS() { #ifdef SUPPORTS_IGES m_fmtIGES = false; #endif m_overwrite = false; m_useGridOrigin = false; m_useDrillOrigin = false; m_includeVirtual = true; m_substModels = false; m_xOrigin = 0.0; m_yOrigin = 0.0; m_minDistance = MIN_DISTANCE; } KICAD2STEP_FRAME::KICAD2STEP_FRAME( const wxString& title ) : KICAD2STEP_FRAME_BASE( NULL, wxID_ANY, title ) { } void KICAD2STEP_FRAME::OnOKButtonClick( wxCommandEvent& aEvent ) { Close(); } PANEL_KICAD2STEP::PANEL_KICAD2STEP( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style ) : wxPanel( parent, id, pos, size, style ), m_error( false ), m_fail( false ) { wxBoxSizer* bSizer = new wxBoxSizer( wxVERTICAL ); m_tcMessages = new wxTextCtrl( this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE | wxTE_READONLY ); bSizer->Add( m_tcMessages, 1, wxALL | wxEXPAND, 5 ); SetSizer( bSizer ); Layout(); bSizer->Fit( this ); } void PANEL_KICAD2STEP::AppendMessage( const wxString& aMessage ) { m_tcMessages->AppendText( aMessage ); wxSafeYield(); } int PANEL_KICAD2STEP::RunConverter() { wxFileName fname( m_params.m_filename ); if( !fname.FileExists() ) { wxMessageBox( wxString::Format( _( "No such file: %s" ), m_params.m_filename ) ); return -1; } wxFileName out_fname; if( m_params.m_outputFile.empty() ) { out_fname.Assign( fname.GetFullPath() ); out_fname.SetExt( m_params.getOutputExt() ); } else { out_fname.Assign( m_params.m_outputFile ); // Set the file extension if the user's requested // file name does not have an extension. if( !out_fname.HasExt() ) out_fname.SetExt( m_params.getOutputExt() ); } if( out_fname.FileExists() && !m_params.m_overwrite ) { ReportMessage( _( "** Output already exists. Export aborted. **\n" "Enable the force overwrite flag to overwrite it." ) ); return -1; } wxString outfile = out_fname.GetFullPath(); KICADPCB pcb( fname.GetName() ); pcb.SetOrigin( m_params.m_xOrigin, m_params.m_yOrigin ); pcb.SetMinDistance( m_params.m_minDistance ); ReportMessage( wxString::Format( _( "Read file: '%s'\n" ), m_params.m_filename ) ); Message::DefaultMessenger()->RemovePrinters( STANDARD_TYPE( Message_PrinterOStream ) ); Message::DefaultMessenger()->AddPrinter( new KiCadPrinter ); if( pcb.ReadFile( m_params.m_filename ) ) { if( m_params.m_useDrillOrigin ) pcb.UseDrillOrigin( true ); if( m_params.m_useGridOrigin ) pcb.UseGridOrigin( true ); bool res; try { ReportMessage( _( "Build STEP data\n" ) ); res = pcb.ComposePCB( m_params.m_includeVirtual, m_params.m_substModels ); if( !res ) { ReportMessage( _( "\n** Error building STEP board model. Export aborted. **\n" ) ); return -1; } ReportMessage( _( "Write STEP file\n" ) ); #ifdef SUPPORTS_IGES if( m_fmtIGES ) res = pcb.WriteIGES( outfile ); else #endif res = pcb.WriteSTEP( outfile ); if( !res ) { ReportMessage( _( "\n** Error writing STEP file. **\n" ) ); return -1; } else { ReportMessage( wxString::Format( _( "\nSTEP file '%s' created.\n" ), outfile ) ); } } catch( const Standard_Failure& e ) { ReportMessage( e.GetMessageString() ); ReportMessage( _( "\n** Error exporting STEP file. Export aborted. **\n" ) ); return -1; } catch( ... ) { ReportMessage( _( "\n** Error exporting STEP file. Export aborted. **\n" ) ); return -1; } } else { ReportMessage( _( "\n** Error reading kicad_pcb file. **\n" ) ); return -1; } wxString msg; if( m_fail ) { msg = _( "Unable to create STEP file.\n" "Check that the board has a valid outline and models." ); } else if( m_error ) { msg = _( "STEP file has been created, but there are warnings." ); } ReportMessage( msg ); return 0; } wxString KICAD2MCAD_PRMS::getOutputExt() const { #ifdef SUPPORTS_IGES if( m_fmtIGES ) return wxT( "igs" ); else #endif return wxT( "step" ); } KICAD2STEP::KICAD2STEP( KICAD2MCAD_PRMS aParams ) : m_params( aParams ), m_panel( nullptr ) { } int KICAD2STEP::Run() { // create the main application window KICAD2STEP_FRAME* frame = new KICAD2STEP_FRAME( wxT( "Kicad2step" ) ); m_panel = frame->m_panelKicad2Step; m_panel->m_params = m_params; // and show it (a wxFrame is not shown when created initially) frame->Show( true ); frame->Iconize( false ); openPanel = m_panel; int diag = m_panel->RunConverter(); openPanel = nullptr; return diag; } void KICAD2STEP::ReportMessage( const wxString& aMessage ) { m_panel->AppendMessage( aMessage ); }