diff --git a/utils/kicad2step/pcb/kicadcurve.cpp b/utils/kicad2step/pcb/kicadcurve.cpp index 8171bffa0d..358d03ea85 100644 --- a/utils/kicad2step/pcb/kicadcurve.cpp +++ b/utils/kicad2step/pcb/kicadcurve.cpp @@ -2,6 +2,7 @@ * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2016 Cirilo Bernardo <cirilo.bernardo@gmail.com> + * Copyright (C) 2022 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 @@ -53,6 +54,7 @@ KICADCURVE::~KICADCURVE() } #include <sexpr/sexpr_parser.h> +#include <geometry/shape_line_chain.h> bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) { @@ -60,7 +62,7 @@ bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) && CURVE_CIRCLE != aCurveType && CURVE_BEZIER != aCurveType && CURVE_POLYGON != aCurveType ) { - wxLogMessage( wxT( "* Unsupported curve type: %d\n" ), aCurveType ); + wxLogMessage( "* Unsupported curve type: %d\n", aCurveType ); return false; } @@ -74,7 +76,7 @@ bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) || ( CURVE_BEZIER == aCurveType && nchild < 5 ) || ( CURVE_POLYGON == aCurveType && nchild < 5 ) ) { - wxLogMessage( wxT( "* bad curve data; not enough parameters\n" ) ); + wxLogMessage( "* bad curve data; not enough parameters\n" ); return false; } @@ -110,6 +112,8 @@ bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) SEXPR::SEXPR* sub_child = ( *list )[ii]; text = sub_child->GetChild( 0 )->GetSymbol(); + // inside pts list, parmeters are xy point coord + // or a arc (start, middle, end) points if( text == "xy" ) { DOUBLET coord; @@ -132,6 +136,49 @@ bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) else m_poly.push_back( coord ); } + else if( text == "arc" ) + { + int arc_child = sub_child->GetNumberOfChildren(); + DOUBLET arc_start, arc_middle, arc_end; + + for( int jj = 1; jj < arc_child; jj++ ) + { + SEXPR::SEXPR* curr_child = sub_child->GetChild( jj ); + text = curr_child->GetChild( 0 )->GetSymbol(); + + if( text == "start" ) + { + if( !Get2DCoordinate( curr_child, arc_start ) ) + return false; + } + else if( text == "end" ) + { + if( !Get2DCoordinate( curr_child, arc_end ) ) + return false; + } + else if( text == "mid" ) + { + if( !Get2DCoordinate( curr_child, arc_middle ) ) + return false; + } + } + + // To convert arc edge to segments, we are using SHAPE_ARC, but SHAPE_ARC use + // integer coords. So to avoid truncations, use a scaling factor. + // 1e5 is enough. + const double scale = 1e5; + SHAPE_ARC new_arc( VECTOR2I( arc_start.x*scale, arc_start.y*scale ), + VECTOR2I( arc_middle.x*scale, arc_middle.y*scale ), + VECTOR2I( arc_end.x*scale, arc_end.y*scale ), 0 ); + + double accuracy = 0.005*scale; // Approx accuracy is 5 microns + SHAPE_LINE_CHAIN segs_from_arc = new_arc.ConvertToPolyline( accuracy ); + + // Add segments to STEP polygon + for( int ll = 0; ll < segs_from_arc.PointCount(); ll++ ) + m_poly.emplace_back( segs_from_arc.CPoint(ll).x/scale, + segs_from_arc.CPoint(ll).y/scale ); + } } } else if( text == "start" || text == "center" ) @@ -154,9 +201,10 @@ bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) else if( text == "angle" ) { if( child->GetNumberOfChildren() < 2 - || ( !child->GetChild( 1 )->IsDouble() && !child->GetChild( 1 )->IsInteger() ) ) + || ( !child->GetChild( 1 )->IsDouble() + && !child->GetChild( 1 )->IsInteger() ) ) { - wxLogMessage( wxT( "* bad angle data\n" ) ); + wxLogMessage( "* bad angle data\n" ); return false; } @@ -175,7 +223,7 @@ bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) { std::ostringstream ostr; ostr << "* bad layer data: " << child->AsString(); - wxLogMessage( wxT( "%s\n" ), ostr.str().c_str() ); + wxLogMessage( "%s\n", ostr.str().c_str() ); return false; } @@ -205,7 +253,7 @@ bool KICADCURVE::Read( SEXPR::SEXPR* aEntry, CURVE_TYPE aCurveType ) m_end.y = new_arc.GetP0().y/scale; m_ep.x = new_arc.GetP1().x/scale; m_ep.y = new_arc.GetP1().y/scale; - m_angle = new_arc.GetCentralAngle().AsRadians(); + m_angle = new_arc.GetCentralAngle() / 180.0 * M_PI; } return true;