202#include <fmt/format.h>
224 m_arcTargetChordLength( 0 ),
227 m_useUserCoords( false ),
228 m_fitUserCoords( false ),
229 m_current_item( nullptr )
238 double aScale,
bool aMirror )
266 fmt::print(
m_outputFile,
"PT {:.1f};\n", penThicknessMM );
290 bbox.
Merge( item.bbox );
292 fmt::print(
m_outputFile,
"SC{:.0f},{:.0f},{:.0f},{:.0f};\n",
301 fmt::print(
m_outputFile,
"SC{:.0f},{:.0f},{:.0f},{:.0f};\n",
316 if( item.loc_start != loc || pen_up )
324 fmt::print(
m_outputFile,
"PA {:.0f},{:.0f};", item.loc_start.x, item.loc_start.y );
327 if( item.dashType != current_dash )
329 current_dash = item.dashType;
333 if( item.pen != current_pen )
342 current_pen = item.pen;
345 if( pen_up && !item.lift_before )
350 else if( !pen_up && item.lift_before )
358 if( !item.pen_returns )
364 if( item.lift_after )
396 std::vector<VECTOR2I> cornerList;
398 cornerList.emplace_back( p1.
x, p1.
y );
399 cornerList.emplace_back( p2.
x, p1.
y );
400 cornerList.emplace_back( p2.
x, p2.
y );
401 cornerList.emplace_back( p1.
x, p2.
y );
402 cornerList.emplace_back( p1.
x, p1.
y );
404 PlotPoly( cornerList, aFill, aWidth,
nullptr );
415 double const circumf = 2.0 * M_PI *
radius;
421 if( aFill == FILL_T::FILLED_SHAPE )
452 if( aFill == FILL_T::NO_FILL && aWidth <= 0 )
455 if( aCornerList.size() <= 1 )
469 if( aFill != FILL_T::NO_FILL )
476 for(
unsigned ii = 1; ii < aCornerList.size(); ++ii )
477 LineTo( aCornerList[ii] );
479 int ii = aCornerList.size() - 1;
481 if( aCornerList[ii] != aCornerList[0] )
487 else if( aWidth != 0 )
490 for(
unsigned ii = 1; ii < aCornerList.size(); ii++ )
491 LineTo( aCornerList[ii] );
494 if( aFill != FILL_T::NO_FILL )
496 int ii = aCornerList.size() - 1;
498 if( aCornerList[ii] != aCornerList[0] )
526 else if( plume ==
'D' )
529 startOrAppendItem( lastpos_dev, fmt::format(
"PA {:.0f},{:.0f};", pos_dev.
x, pos_dev.
y ) );
572 polyArc( aCenter, aStartAngle, aAngle, aRadius, aFill, aWidth );
577 double const circumf_device = 2.0 * M_PI * radius_device;
593 KiROUND( aCenter.
y - aRadius * startAngle.
Sin() ) );
604 VECTOR2D( radius_device * 2, radius_device * 2 ) ) );
619 if( size.
x > size.
y )
621 std::swap( size.
x, size.
y );
625 if( aTraceMode ==
FILLED )
627 int deltaxy = size.
y - size.
x;
652 int radius = diametre / 2;
654 if( trace_mode ==
FILLED )
666 if( trace_mode ==
FILLED )
674 fmt::format(
"PM 0; PA {:.0f},{:.0f};CI {:.0f};{}",
695 std::vector<VECTOR2I> corners;
697 int dx = aPadSize.
x / 2;
698 int dy = aPadSize.
y / 2;
700 if( aTraceMode ==
FILLED )
705 dx = std::max( dx, 0);
707 dy = std::max( dy, 0);
711 corners.emplace_back( - dx, - dy );
712 corners.emplace_back( - dx, + dy );
713 corners.emplace_back( + dx, + dy );
714 corners.emplace_back( + dx, - dy );
717 corners.emplace_back( - dx, - dy );
719 for(
unsigned ii = 0; ii < corners.size(); ii++ )
725 PlotPoly( corners, aTraceMode ==
FILLED ? FILL_T::FILLED_SHAPE : FILL_T::NO_FILL );
730 int aCornerRadius,
const EDA_ANGLE& aOrient,
737 if( aTraceMode ==
FILLED )
741 size.
x = std::max( size.
x, 0);
743 size.
y = std::max( size.
y, 0);
746 aCornerRadius = std::min( aCornerRadius, std::min( size.
x, size.
y ) /2 );
753 std::vector<VECTOR2I> cornerList;
757 for(
int ii = 0; ii < poly.
PointCount(); ++ii )
758 cornerList.emplace_back( poly.
CPoint( ii ).
x, poly.
CPoint( ii ).
y );
760 if( cornerList.back() != cornerList.front() )
761 cornerList.push_back( cornerList.front() );
763 PlotPoly( cornerList, aTraceMode ==
FILLED ? FILL_T::FILLED_SHAPE : FILL_T::NO_FILL );
771 std::vector<VECTOR2I> cornerList;
773 for(
int cnt = 0; cnt < aPolygons->
OutlineCount(); ++cnt )
780 for(
int ii = 0; ii < poly.
PointCount(); ++ii )
781 cornerList.emplace_back( poly.
CPoint( ii ).
x, poly.
CPoint( ii ).
y );
783 if( cornerList.back() != cornerList.front() )
784 cornerList.push_back( cornerList.front() );
786 PlotPoly( cornerList, aTraceMode ==
FILLED ? FILL_T::FILLED_SHAPE : FILL_T::NO_FILL );
795 std::vector<VECTOR2I> cornerList;
796 cornerList.reserve( 5 );
798 for(
int ii = 0; ii < 4; ii++ )
803 cornerList.push_back( coord );
807 cornerList.push_back( cornerList.front() );
809 PlotPoly( cornerList, aTraceMode ==
FILLED ? FILL_T::FILLED_SHAPE : FILL_T::NO_FILL );
859 if( items.size() < 2 )
862 std::list<HPGL_ITEM> target;
885 target.emplace_back( last_item );
887 while( !items.empty() )
889 auto best_it = items.begin();
892 for(
auto search_it = best_it; search_it != items.end(); search_it++ )
895 if( best_it->pen != last_item.
pen && search_it->pen == last_item.
pen )
903 if( dist < best_dist )
911 target.emplace_back( *best_it );
912 last_item = *best_it;
913 items.erase( best_it );
916 items.splice( items.begin(), target );
924 case LINE_STYLE::DASH:
return "LT 2 4 1;";
925 case LINE_STYLE::DOT:
return "LT 1 1 1;";
926 case LINE_STYLE::DASHDOT:
return "LT 4 6 1;";
927 case LINE_STYLE::DASHDOTDOT:
return "LT 7 8 1;";
928 default:
return "LT;";
936 return sqrt( diff.
x * diff.
x + diff.
y * diff.
y );
static std::string hpgl_end_polygon_cmd
static const double PLUsPERDECIMIL
static double dpoint_dist(const VECTOR2D &a, const VECTOR2D &b)
Compute the distance between two VECTOR2D points.
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
constexpr coord_type GetY() const
constexpr size_type GetWidth() const
constexpr coord_type GetX() const
constexpr BOX2< Vec > & Merge(const BOX2< Vec > &aRect)
Modify the position and size of the rectangle in order to contain aRect.
constexpr size_type GetHeight() const
virtual void FlashPadRoundRect(const VECTOR2I &aPadPos, const VECTOR2I &aSize, int aCornerRadius, const EDA_ANGLE &aOrient, OUTLINE_MODE aTraceMode, void *aData) override
HPGL_ITEM * m_current_item
virtual void SetDash(int aLineWidth, LINE_STYLE aLineStyle) override
HPGL supports dashed lines.
virtual void SetPenSpeed(int speed)
bool startItem(const VECTOR2D &location)
Start a new HPGL_ITEM if necessary, keeping the current one if it exists.
double m_arcTargetChordLength
void flushItem()
Flush the current HPGL_ITEM and clear out the current item pointer.
virtual void FlashPadTrapez(const VECTOR2I &aPadPos, const VECTOR2I *aCorners, const EDA_ANGLE &aPadOrient, OUTLINE_MODE aTraceMode, void *aData) override
Flash a trapezoidal pad.
void SetTargetChordLength(double chord_len)
Set the target length of chords used to draw approximated circles and arcs.
virtual void SetPenNumber(int number)
static void sortItems(std::list< HPGL_ITEM > &items)
Sort a list of HPGL items to improve plotting speed on mechanical plotters.
virtual void FlashPadCircle(const VECTOR2I &aPadPos, int aDiameter, OUTLINE_MODE aTraceMode, void *aData) override
virtual bool StartPlot(const wxString &aPageNumber) override
At the start of the HPGL plot pen speed and number are requested.
EDA_ANGLE m_arcMinChordDegrees
virtual void SetCurrentLineWidth(int width, void *aData=nullptr) override
HPGL doesn't handle line thickness or color.
virtual void FlashRegularPolygon(const VECTOR2I &aShapePos, int aDiameter, int aCornerCount, const EDA_ANGLE &aOrient, OUTLINE_MODE aTraceMode, void *aData) override
Flash a regular polygon.
virtual void PlotPoly(const std::vector< VECTOR2I > &aCornerList, FILL_T aFill, int aWidth=USE_DEFAULT_LINE_WIDTH, void *aData=nullptr) override
Draw a polygon ( filled or not ).
static const char * lineStyleCommand(LINE_STYLE aLineStyle)
Return the plot command corresponding to a line type.
virtual void Rect(const VECTOR2I &p1, const VECTOR2I &p2, FILL_T aFill, int aWidth=USE_DEFAULT_LINE_WIDTH) override
virtual void SetViewport(const VECTOR2I &aOffset, double aIusPerDecimil, double aScale, bool aMirror) override
Set the plot offset and scaling for the current plot.
std::list< HPGL_ITEM > m_items
virtual void FlashPadCustom(const VECTOR2I &aPadPos, const VECTOR2I &aSize, const EDA_ANGLE &aOrient, SHAPE_POLY_SET *aPolygons, OUTLINE_MODE aTraceMode, void *aData) override
virtual void SetPenDiameter(double diameter)
virtual void FlashPadOval(const VECTOR2I &aPadPos, const VECTOR2I &aSize, const EDA_ANGLE &aPadOrient, OUTLINE_MODE aTraceMode, void *aData) override
bool startOrAppendItem(const VECTOR2D &location, const std::string &content="")
Start a new HPGL_ITEM with the given string if necessary, or append the string to the current item.
virtual void Arc(const VECTOR2D &aCenter, const EDA_ANGLE &aStartAngle, const EDA_ANGLE &aAngle, double aRadius, FILL_T aFill, int aWidth=USE_DEFAULT_LINE_WIDTH) override
Plot an arc.
virtual void PenTo(const VECTOR2I &pos, char plume) override
Moveto/lineto primitive, moves the 'pen' to the specified direction.
virtual void ThickSegment(const VECTOR2I &start, const VECTOR2I &end, int width, OUTLINE_MODE tracemode, void *aData) override
virtual void Circle(const VECTOR2I &aCenter, int aDiameter, FILL_T aFill, int aWidth=USE_DEFAULT_LINE_WIDTH) override
virtual bool EndPlot() override
HPGL end of plot: sort and emit graphics, pen return and release.
virtual void FlashPadRect(const VECTOR2I &aPadPos, const VECTOR2I &aSize, const EDA_ANGLE &aOrient, OUTLINE_MODE aTraceMode, void *aData) override
const VECTOR2D & GetSizeMils() const
static const int USE_DEFAULT_LINE_WIDTH
virtual void polyArc(const VECTOR2D &aCentre, const EDA_ANGLE &aStartAngle, const EDA_ANGLE &aAngle, double aRadius, FILL_T aFill, int aWidth=USE_DEFAULT_LINE_WIDTH)
Generic fallback: arc rendered as a polyline.
void MoveTo(const VECTOR2I &pos)
void FinishTo(const VECTOR2I &pos)
virtual VECTOR2D userToDeviceCoordinates(const VECTOR2I &aCoordinate)
Modify coordinates according to the orientation, scale factor, and offsets trace.
virtual VECTOR2D userToDeviceSize(const VECTOR2I &size)
Modify size according to the plotter scale factors (VECTOR2I version, returns a VECTOR2D).
void sketchOval(const VECTOR2I &aPos, const VECTOR2I &aSize, const EDA_ANGLE &aOrient, int aWidth)
int GetPlotterArcHighDef() const
double m_plotScale
Plot scale - chosen by the user (even implicitly with 'fit in a4')
FILE * m_outputFile
Output file.
void LineTo(const VECTOR2I &pos)
void segmentAsOval(const VECTOR2I &start, const VECTOR2I &end, int width, OUTLINE_MODE tracemode)
Convert a thick segment and plot it as an oval.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
int PointCount() const
Return the number of points (vertices) in this line chain.
const VECTOR2I & CPoint(int aIndex) const
Return a reference to a given point in the line chain.
Represent a set of closed polygons.
SHAPE_LINE_CHAIN & Outline(int aIndex)
Return the reference to aIndex-th outline in the set.
int OutlineCount() const
Return the number of outlines in the set.
void TransformRoundChamferedRectToPolygon(SHAPE_POLY_SET &aBuffer, const VECTOR2I &aPosition, const VECTOR2I &aSize, const EDA_ANGLE &aRotation, int aCornerRadius, double aChamferRatio, int aChamferCorners, int aInflate, int aError, ERROR_LOC aErrorLoc)
Convert a rectangle with rounded corners and/or chamfered corners to a polygon.
static constexpr EDA_ANGLE ANGLE_90
static constexpr EDA_ANGLE ANGLE_45
static constexpr EDA_ANGLE ANGLE_360
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
LINE_STYLE
Dashed line types.
bool pen_returns
Whether the pen returns to its original state after the command.
BOX2D bbox
Bounding box of this item.
LINE_STYLE dashType
Pen number for this command.
bool lift_after
Whether the pen must be lifted after the command.
std::string content
Line style for this command.
bool lift_before
Whether the command should be executed with the pen lifted.
VECTOR2D loc_start
Location the pen should start at.
VECTOR2D loc_end
Location the pen will be at when it finishes.
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.
VECTOR2< int32_t > VECTOR2I
VECTOR2< double > VECTOR2D