87#define MIN_BULGE 0.0218
89#define SCALE_FACTOR(x) (x)
101 m_maxX =
m_maxY = std::numeric_limits<double>::lowest();
109 m_layers.push_back( std::move( layer0 ) );
126 catch(
const std::bad_alloc& )
134 ReportMsg(
_(
"Memory was exhausted trying to load the DXF, it may be too large." ) );
146 catch(
const std::bad_alloc& )
154 ReportMsg(
_(
"Memory was exhausted trying to load the DXF, it may be too large." ) );
223 FILE* fp = wxFopen( aFile, wxT(
"rt" ) );
230 bool success = dxf_reader.in( fp,
this );
240 std::string str(
reinterpret_cast<char*
>( aMemBuffer.GetData() ), aMemBuffer.GetDataLen() );
244 bool success = dxf_reader.in( str,
this );
280 m_curr_entity.m_SplineControlPointList.emplace_back( aData.x , aData.y, aData.w );
288 m_curr_entity.m_SplineFitPointList.emplace_back( aData.x, aData.y );
301 wxString
name = wxString::FromUTF8( aData.name.c_str() );
303 int lw = attributes.getWidth();
308 std::unique_ptr<DXF_IMPORT_LAYER> layer = std::make_unique<DXF_IMPORT_LAYER>(
name, lw );
310 m_layers.push_back( std::move( layer ) );
318 wxString description =
From_UTF8( data.description.c_str() );
341 wxString layerName = wxString::FromUTF8( aLayerName.c_str() );
343 if( !layerName.IsEmpty() )
346 [layerName](
const auto& it )
348 return it->m_layerName == layerName;
352 layer = resultIt->get();
361 wxString layerName = wxString::FromUTF8( aLayerName.c_str() );
363 if( layerName.IsEmpty() )
364 layerName = wxS(
"0" );
373 wxString blockName = wxString::FromUTF8( aBlockName.c_str() );
375 if( !blockName.IsEmpty() )
378 [blockName](
const auto& it )
380 return it->m_name == blockName;
384 block = resultIt->get();
394 wxString styleName = wxString::FromUTF8( aStyleName.c_str() );
396 if( !styleName.IsEmpty() )
399 [styleName](
const auto& it )
401 return it->m_name == styleName;
405 style = resultIt->get();
424 bufferToUse->
AddLine( start,
end, lineWidth );
457 if( aData.startWidth > 0.0 )
458 lineWidth = aData.startWidth / 100.0;
459 else if ( aData.endWidth > 0.0 )
460 lineWidth = aData.endWidth / 100.0;
462 const DL_VertexData* vertex = &aData;
523 wxString
name = wxString::FromUTF8( aData.name.c_str() );
525 std::unique_ptr<DXF_IMPORT_BLOCK> block = std::make_unique<DXF_IMPORT_BLOCK>(
name, aData.bpx,
528 m_blocks.push_back( std::move( block ) );
543 if( block ==
nullptr )
562 for(
const std::unique_ptr<IMPORTED_SHAPE>& shape : block->
m_buffer.
GetShapes() )
564 std::unique_ptr<IMPORTED_SHAPE> newShape = shape->clone();
566 newShape->Transform( trans, translation );
568 if( newShape->GetSourceLayer().IsEmpty() || newShape->GetSourceLayer() == wxS(
"0" ) )
569 newShape->SetSourceLayer( insertLayer );
614 std::swap( startangle, endangle );
618 VECTOR2D startPoint( aData.radius, 0.0 );
621 mapY( startPoint.
y + centerCoords.
y ) );
624 EDA_ANGLE angle = -( endangle - startangle );
636 bufferToUse->
AddArc(
center, arcStart, angle, lineWidth );
667 if( startAngle > endAngle )
670 if( aData.ratio == 1.0 )
674 if( startAngle == endAngle )
676 DL_CircleData
circle( aData.cx, aData.cy, aData.cz,
radius );
686 DL_ArcData arc( aData.cx, aData.cy, aData.cz,
radius, startAngle.
AsDegrees(),
704 double minorRadius = majorRadius * aData.ratio;
707 if( startAngle == endAngle ||
std::abs( ( endAngle - startAngle ).AsDegrees() - 360.0 ) < 1e-9 )
709 bufferToUse->
AddEllipse(
center, majorRadius, minorRadius, rotation, lineWidth,
false );
713 bufferToUse->
AddEllipseArc(
center, majorRadius, minorRadius, rotation, startAngle, endAngle, lineWidth );
728 std::isnan( aData.apy ) ? 0 : aData.apy,
729 std::isnan( aData.apz ) ? 0 : aData.apz ) );
734 if( aData.vJustification != 0 || aData.hJustification != 0 || aData.hJustification == 4 )
736 if( aData.hJustification != 3 && aData.hJustification != 5 )
748 double textHeight =
mapDim( aData.height );
751 double charWidth = textHeight * 0.9;
753 if( style !=
nullptr )
756 double textWidth = charWidth *
text.length();
757 double textThickness = textHeight / 8.0;
767 switch( aData.vJustification )
773 topLeft.
y = textHeight;
774 topRight.
y = textHeight;
780 bottomRight.
y = -textHeight / 2.0;
781 bottomLeft.
y = -textHeight / 2.0;
782 topLeft.
y = textHeight / 2.0;
783 topRight.
y = textHeight / 2.0;
789 bottomLeft.
y = -textHeight;
790 bottomRight.
y = -textHeight;
794 switch( aData.hJustification )
801 bottomRight.
x = textWidth;
802 topRight.
x = textWidth;
809 bottomLeft.
x = -textWidth / 2.0;
810 topLeft.
x = -textWidth / 2.0;
811 bottomRight.
x = textWidth / 2.0;
812 topRight.
x = textWidth / 2.0;
818 bottomLeft.
x = -textWidth;
819 topLeft.
x = -textWidth;
824 wxString sty = wxString::FromUTF8( aData.style.c_str() );
827 if( aData.textgen == 2 )
830 }
else if( aData.textgen == 4 )
840 double angle_degree = aData.angle * 180 /
M_PI;
844 double angleInRads = angle_degree *
M_PI / 180.0;
845 double cosine = cos(angleInRads);
846 double sine = sin(angleInRads);
851 bufferToUse->
AddText( refPoint,
text, textHeight, charWidth, textThickness, angle_degree,
852 hJustify, vJustify );
855 bottomLeft.
x = bottomLeft.
x * cosine - bottomLeft.
y * sine;
856 bottomLeft.
y = bottomLeft.
x * sine + bottomLeft.
y * cosine;
858 bottomRight.
x = bottomRight.
x * cosine - bottomRight.
y * sine;
859 bottomRight.
y = bottomRight.
x * sine + bottomRight.
y * cosine;
861 topLeft.
x = topLeft.
x * cosine - topLeft.
y * sine;
862 topLeft.
y = topLeft.
x * sine + topLeft.
y * cosine;
864 topRight.
x = topRight.
x * cosine - topRight.
y * sine;
865 topRight.
y = topRight.
x * sine + topRight.
y * cosine;
867 bottomLeft += refPoint;
868 bottomRight += refPoint;
870 topRight += refPoint;
897 double textHeight =
mapDim( aData.height );
900 double charWidth = textHeight * 0.9;
902 if( style !=
nullptr )
905 double textWidth = charWidth *
text.length();
906 double textThickness = textHeight/8.0;
921 if( aData.attachmentPoint <= 3 )
925 bottomLeft.
y = -textHeight;
926 bottomRight.
y = -textHeight;
928 else if( aData.attachmentPoint <= 6 )
932 bottomRight.
y = -textHeight / 2.0;
933 bottomLeft.
y = -textHeight / 2.0;
934 topLeft.
y = textHeight / 2.0;
935 topRight.
y = textHeight / 2.0;
941 topLeft.
y = textHeight;
942 topRight.
y = textHeight;
945 if( aData.attachmentPoint % 3 == 1 )
949 bottomRight.
x = textWidth;
950 topRight.
x = textWidth;
952 else if( aData.attachmentPoint % 3 == 2 )
956 bottomLeft.
x = -textWidth / 2.0;
957 topLeft.
x = -textWidth / 2.0;
958 bottomRight.
x = textWidth / 2.0;
959 topRight.
x = textWidth / 2.0;
965 bottomLeft.
x = -textWidth;
966 topLeft.
x = -textWidth;
970 if( data.alignH == 1 )
974 else if( data.alignH == 3 )
983 if( aData.alignV == 1 )
995 double angle_degree = aData.angle * 180/
M_PI;
999 double angleInRads = angle_degree *
M_PI / 180.0;
1000 double cosine = cos(angleInRads);
1001 double sine = sin(angleInRads);
1007 bufferToUse->
AddText( textpos,
text, textHeight, charWidth, textThickness, angle_degree,
1008 hJustify, vJustify );
1010 bottomLeft.
x = bottomLeft.
x * cosine - bottomLeft.
y * sine;
1011 bottomLeft.
y = bottomLeft.
x * sine + bottomLeft.
y * cosine;
1013 bottomRight.
x = bottomRight.
x * cosine - bottomRight.
y * sine;
1014 bottomRight.
y = bottomRight.
x * sine + bottomRight.
y * cosine;
1016 topLeft.
x = topLeft.
x * cosine - topLeft.
y * sine;
1017 topLeft.
y = topLeft.
x * sine + topLeft.
y * cosine;
1019 topRight.
x = topRight.
x * cosine - topRight.
y * sine;
1020 topRight.
y = topRight.
x * sine + topRight.
y * cosine;
1022 bottomLeft += textpos;
1023 bottomRight += textpos;
1025 topRight += textpos;
1077 if( key ==
"$DWGCODEPAGE" )
1083 if( key ==
"$AUPREC" )
1089 if( key ==
"$LUPREC" )
1095 if( key ==
"$INSUNITS" )
1145 for(
unsigned i = 0; i<aStr.length(); ++i )
1149 if( c > 175 || c < 11 )
1151 res.append( aStr.Mid( j, i - j ) );
1157 res += wxT(
"\\P" );
1167 res += wxT(
"%%C" );
1172 res += wxT(
"%%D" );
1177 res += wxT(
"%%P" );
1189 res.append( aStr.Mid( j ) );
1199 int overbarLevel = -1;
1205 for( i = 0; i < aData.length(); i++ )
1207 switch( (
wchar_t) aData[i] )
1214 if( overbarLevel == braces )
1223 if( ++i >= aData.length() )
1226 switch( (
wchar_t) aData[i] )
1228 case 'I':
res <<
'\t';
break;
1229 case 'J':
res <<
'\b';
break;
1230 case ' ':
res <<
'^';
break;
1237 if( ++i >= aData.length() )
1240 switch( (
wchar_t) aData[i] )
1256 for( ; codeHex.length() < 4 && i < aData.length(); i++ )
1257 codeHex << aData[i];
1259 unsigned long codeVal = 0;
1261 if( codeHex.ToCULong( &codeVal, 16 ) && codeVal != 0 )
1262 res << wxUniChar( codeVal );
1273 for( ; i < aData.length(); i++ )
1275 if( aData[i] ==
';' )
1278 stacked << aData[i];
1281 if( stacked.Contains( wxS(
"#" ) ) )
1284 res << stacked.BeforeFirst(
'#' );
1285 res <<
'}' <<
'/' <<
'_' <<
'{';
1286 res << stacked.AfterFirst(
'#' );
1291 stacked.Replace( wxS(
"^ " ), wxS(
"/" ) );
1298 if( overbarLevel == -1 )
1301 overbarLevel = braces;
1305 if( overbarLevel == braces )
1331 for( ; i < aData.length(); i++ )
1333 if( aData[i] ==
';' )
1339 if( ++i >= aData.length() )
1348 default:
res << aData[i];
1352 if( overbarLevel != -1 )
1362 regexp.Compile( wxT(
"%%[cC]" ) );
1365 regexp.Replace( &
res, wxChar( 0xD8 ) );
1368 regexp.Replace( &
res, wxChar( 0x2205 ) );
1372 regexp.Compile( wxT(
"%%[dD]" ) );
1373 regexp.Replace( &
res, wxChar( 0x00B0 ) );
1376 regexp.Compile( wxT(
"%%[pP]" ) );
1377 regexp.Replace( &
res, wxChar( 0x00B1 ) );
1386 wxString
name = wxString::FromUTF8( aData.name.c_str() );
1388 auto style = std::make_unique<DXF_IMPORT_STYLE>(
name, aData.fixedTextHeight, aData.widthFactor,
1389 aData.bold, aData.italic );
1391 m_styles.push_back( std::move( style ) );
1405 double lineWidth = 0.0001;
1406 double thickness =
mapDim( std::max( aData.thickness, 0.01 ) );
1421 const VECTOR2D& aSegEnd,
double aWidth )
1437 double aBulge,
double aWidth )
1443 if( aBulge < -2000.0 )
1445 else if( aBulge > 2000.0 )
1448 double ang = 4.0 * atan( aBulge );
1455 double offAng = atan2( ep.
y - sp.
y, ep.
x - sp.
x );
1458 double d = 0.5 * sqrt( ( sp.
x - ep.
x ) * ( sp.
x - ep.
x ) + ( sp.
y - ep.
y ) * ( sp.
y - ep.
y ) );
1461 double xm = ( sp.
x + ep.
x ) * 0.5;
1462 double ym = ( sp.
y + ep.
y ) * 0.5;
1463 double radius = d / sin( ang * 0.5 );
1475 double h = sqrt( dh2 );
1487 else if( ang >
M_PI )
1491 double cx = h * cos( offAng ) + xm;
1492 double cy = h * sin( offAng ) + ym;
1510 bufferToUse->
AddArc(
center, arc_start, angle, aWidth );
1519#include "tinysplinecxx.h"
1524 wxLogMessage(
"spl deg %d kn %d ctr %d fit %d",
1531 unsigned imax =
m_curr_entity.m_SplineControlPointList.size();
1540 for(
unsigned int ii = 1; ii < imax; ++ii )
1545 if( startpoint != endpoint )
1552 startpoint = endpoint;
1556 std::vector<double> ctrlp;
1558 for(
unsigned ii = 0; ii < imax; ++ii )
1560 ctrlp.push_back(
m_curr_entity.m_SplineControlPointList[ii].m_x );
1561 ctrlp.push_back(
m_curr_entity.m_SplineControlPointList[ii].m_y );
1564 tinyspline::BSpline beziers;
1565 std::vector<double> coords;
1569 tinyspline::BSpline dxfspline(
m_curr_entity.m_SplineControlPointList.size(),
1572 dxfspline.setControlPoints( ctrlp );
1575 if( dxfspline.degree() < 3 )
1576 dxfspline = dxfspline.elevateDegree( 3 - dxfspline.degree() );
1578 beziers = dxfspline.toBeziers();
1579 coords = beziers.controlPoints();
1581 catch(
const std::runtime_error& )
1585 ReportMsg(
_(
"Invalid spline definition encountered" ) );
1589 size_t order = beziers.order();
1590 size_t dim = beziers.dimension();
1591 size_t numBeziers = ( coords.size() / dim ) / order;
1593 for(
size_t i = 0; i < numBeziers; i++ )
1595 size_t ii = i * dim * order;
1602 if( ii + 5 >= coords.size() )
1603 bezierControl2 = bezierControl1;
1605 bezierControl2 =
VECTOR2D(
mapX( coords[ii + 4] ),
mapY( coords[ii + 5] ) );
1609 if( ii + 7 >= coords.size() )
1610 end = bezierControl2;
1617 bufferToUse->
AddSpline( start, bezierControl1, bezierControl2,
end, aWidth );
1637 double direction[3];
1638 aData->getDirection( direction );
1642 if( ( abs( arbZ.
x ) < ( 1.0 / 64.0 ) ) && ( abs( arbZ.
y ) < ( 1.0 / 64.0 ) ) )
1655 return arbitraryAxis * point;
1665 MATRIX3x3 world( worldX, worldY, worldZ );
1667 return world * point;
constexpr void SetOrigin(const Vec &pos)
constexpr void SetEnd(coord_type x, coord_type y)
A helper class to hold layer settings temporarily during import.
GRAPHICS_IMPORTER_BUFFER m_buffer
A helper class to hold layer settings temporarily during import.
DXF_IMPORT_UNITS m_currentUnit
virtual void addPolyline(const DL_PolylineData &aData) override
double mapY(double aDxfCoordY)
std::vector< std::unique_ptr< DXF_IMPORT_LAYER > > m_layers
void addEllipse(const DL_EllipseData &aData) override
static wxString toNativeString(const wxString &aData)
Convert a DXF encoded string into a native Unicode string.
virtual void setVariableInt(const std::string &key, int value, int code) override
Called for every int variable in the DXF file (e.g.
bool LoadFromMemory(const wxMemoryBuffer &aMemBuffer) override
Set memory buffer with content for import.
void insertSpline(double aWidth)
virtual void addLine(const DL_LineData &aData) override
VECTOR3D ocsToWcs(const MATRIX3x3D &arbitraryAxis, VECTOR3D point)
Convert a given object coordinate point to world coordinate using the given arbitrary axis vectors.
void insertLine(const VECTOR2D &aSegStart, const VECTOR2D &aSegEnd, double aWidth)
DXF_IMPORT_BLOCK * getImportBlock(const std::string &aBlockName)
Return the import layer block.
double getCurrentUnitScale()
virtual void addBlock(const DL_BlockData &) override
Called for each BLOCK in the DXF file.
virtual void addLayer(const DL_LayerData &aData) override
virtual void addCircle(const DL_CircleData &aData) override
virtual void addMText(const DL_MTextData &aData) override
DXF_IMPORT_LAYER * getImportLayer(const std::string &aLayerName)
Return the import layer data.
static wxString toDxfString(const wxString &aStr)
Convert a native Unicode string into a DXF encoded string.
DXF_IMPORT_STYLE * getImportStyle(const std::string &aStyleName)
Return the import style.
virtual void addInsert(const DL_InsertData &aData) override
std::string m_mtextContent
BOX2D GetImageBBox() const override
Return image bounding box from original imported file.
MATRIX3x3D getArbitraryAxis(DL_Extrusion *aData)
double GetImageWidth() const override
Return image width from original imported file.
double GetImageHeight() const override
Return image height from original imported file.
void updateImageLimits(const VECTOR2D &aPoint)
virtual void addTextStyle(const DL_StyleData &aData) override
virtual void addVertex(const DL_VertexData &aData) override
Called for every polyline vertex.
VECTOR3D wcsToOcs(const MATRIX3x3D &arbitraryAxis, VECTOR3D point)
Convert a given world coordinate point to object coordinate using the given arbitrary axis vectors.
virtual void addSpline(const DL_SplineData &aData) override
Called for every spline.
virtual void addText(const DL_TextData &aData) override
virtual void endBlock() override
virtual void setVariableString(const std::string &key, const std::string &value, int code) override
Called for every string variable in the DXF file (e.g.
virtual void addKnot(const DL_KnotData &aData) override
Called for every spline knot value.
bool ImportDxfFile(const wxString &aFile)
Implementation of the method used for communicate with this filter.
virtual void addArc(const DL_ArcData &aData) override
void SetDefaultLineWidthMM(double aWidth)
Set the default line width when importing dxf items like lines to Pcbnew.
virtual void addFitPoint(const DL_FitPointData &aData) override
Called for every spline fit point.
virtual void addPoint(const DL_PointData &aData) override
double mapDim(double aDxfValue)
void ReportMsg(const wxString &aMessage) override
int m_importCoordinatePrecision
double mapX(double aDxfCoordX)
double lineWeightToWidth(int lw, DXF_IMPORT_LAYER *aLayer)
void insertArc(const VECTOR2D &aSegStart, const VECTOR2D &aSegEnd, double aBulge, double aWidth)
std::vector< std::unique_ptr< DXF_IMPORT_BLOCK > > m_blocks
bool Import() override
Actually imports the file.
virtual void addMTextChunk(const std::string &text) override
int m_importAnglePrecision
virtual void endEntity() override
virtual void addControlPoint(const DL_ControlPointData &aData) override
Called for every spline control point.
double m_defaultThickness
GRAPHICS_IMPORTER_BUFFER m_internalImporter
std::vector< std::unique_ptr< DXF_IMPORT_STYLE > > m_styles
bool Load(const wxString &aFileName) override
Load file for import.
DXF_IMPORT_BLOCK * m_currentBlock
DXF2BRD_ENTITY_DATA m_curr_entity
wxString getDxfLayerName(const std::string &aLayerName) const
virtual void SetImporter(GRAPHICS_IMPORTER *aImporter) override
Set the receiver of the imported shapes.
virtual void addLinetype(const DL_LinetypeData &data) override
A helper class to hold style settings temporarily during import.
std::list< std::unique_ptr< IMPORTED_SHAPE > > & GetShapes()
void AddCircle(const VECTOR2D &aCenter, double aRadius, const IMPORTED_STROKE &aStroke, bool aFilled, const COLOR4D &aFillColor=COLOR4D::UNSPECIFIED) override
Create an object representing a circle.
void AddSpline(const VECTOR2D &aStart, const VECTOR2D &aBezierControl1, const VECTOR2D &aBezierControl2, const VECTOR2D &aEnd, const IMPORTED_STROKE &aStroke) override
Create an object representing an arc.
void AddLine(const VECTOR2D &aStart, const VECTOR2D &aEnd, const IMPORTED_STROKE &aStroke) override
Create an object representing a line segment.
void AddEllipse(const VECTOR2D &aCenter, double aMajorRadius, double aMinorRadius, const EDA_ANGLE &aRotation, const IMPORTED_STROKE &aStroke, bool aFilled, const COLOR4D &aFillColor=COLOR4D::UNSPECIFIED) override
Create an object representing a closed ellipse.
void SetCurrentSourceLayer(const wxString &aSourceLayer) override
Set the source layer for the next buffered shape to be imported.
void AddArc(const VECTOR2D &aCenter, const VECTOR2D &aStart, const EDA_ANGLE &aAngle, const IMPORTED_STROKE &aStroke) override
Create an object representing an arc.
void AddText(const VECTOR2D &aOrigin, const wxString &aText, double aHeight, double aWidth, double aThickness, double aOrientation, GR_TEXT_H_ALIGN_T aHJustify, GR_TEXT_V_ALIGN_T aVJustify, const COLOR4D &aColor=COLOR4D::UNSPECIFIED) override
Create an object representing a text.
void AddEllipseArc(const VECTOR2D &aCenter, double aMajorRadius, double aMinorRadius, const EDA_ANGLE &aRotation, const EDA_ANGLE &aStartAngle, const EDA_ANGLE &aEndAngle, const IMPORTED_STROKE &aStroke) override
Create an object representing an elliptical arc.
Interface that creates objects representing shapes for a given data model.
GRAPHICS_IMPORTER * m_importer
Importer used to create objects representing the imported shapes.
virtual void SetImporter(GRAPHICS_IMPORTER *aImporter)
Set the receiver of the imported shapes.
MATRIX3x3 describes a general 3x3 matrix.
void SetRotation(T aAngle)
Set the rotation components of the matrix.
VECTOR2< T > GetScale() const
Get the scale components of the matrix.
T EuclideanNorm() const
Compute the Euclidean norm of the vector, which is defined as sqrt(x ** 2 + y ** 2).
VECTOR3< T > Normalize()
Compute the normalized vector.
VECTOR3< T > Cross(const VECTOR3< T > &aVector) const
Compute cross product of self with aVector.
#define DXF_IMPORT_LINEWEIGHT_BY_LW_DEFAULT
#define DXF_IMPORT_LINEWEIGHT_BY_LAYER
static constexpr EDA_ANGLE ANGLE_0
static constexpr EDA_ANGLE ANGLE_360
static constexpr EDA_ANGLE ANGLE_180
This file contains miscellaneous commonly used macros and functions.
MATRIX3x3< double > MATRIX3x3D
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
wxString From_UTF8(const char *cstring)
SHAPE_CIRCLE circle(c.m_circle_center, c.m_circle_radius)
GR_TEXT_H_ALIGN_T
This is API surface mapped to common.types.HorizontalAlignment.
GR_TEXT_V_ALIGN_T
This is API surface mapped to common.types.VertialAlignment.
void RotatePoint(int *pX, int *pY, const EDA_ANGLE &aAngle)
Calculate the new point of coord coord pX, pY, for a rotation center 0, 0.
double DEG2RAD(double deg)
VECTOR2< double > VECTOR2D
VECTOR3< double > VECTOR3D