/** * Common plot library \n * Plot settings, postscript plotting, gerber plotting. * * @file plot_common.h */ #ifndef PLOT_COMMON_H_ #define PLOT_COMMON_H_ #include #include "drawtxt.h" #include "common.h" // PAGE_INFO /** * Enum PlotFormat * is the set of supported output plot formats. They should be kept in order * of the radio buttons in the plot panel/windows. */ enum PlotFormat { PLOT_FORMAT_HPGL, PLOT_FORMAT_GERBER, PLOT_FORMAT_POST, PLOT_FORMAT_DXF }; class PLOTTER { public: PLOTTER( PlotFormat aPlotType ); virtual ~PLOTTER() { // Emergency cleanup if( output_file ) { fclose( output_file ); } } /** * Function GetPlotterType * @return the format of the plot file */ PlotFormat GetPlotterType() { return m_PlotType; } virtual bool start_plot( FILE* fout ) = 0; virtual bool end_plot() = 0; virtual void set_negative( bool _negative ) { negative_mode = _negative; } virtual void set_color_mode( bool _color_mode ) { color_mode = _color_mode; } bool get_color_mode() const { return color_mode; } void SetPageSettings( const PAGE_INFO& aPageSettings ); virtual void set_current_line_width( int width ) = 0; virtual void set_default_line_width( int width ) = 0; virtual void set_color( int color ) = 0; virtual void set_dash( bool dashed ) = 0; virtual int get_current_line_width() { return current_pen_width; } virtual void set_plot_width_adj( double width ) { } virtual double get_plot_width_adj() { return 0.; } virtual void set_creator( const wxString& _creator ) { creator = _creator; } virtual void set_filename( const wxString& _filename ) { filename = _filename; } /// Set the plot offset for the current plotting virtual void set_viewport( wxPoint aOffset, double aScale, bool aMirror ) = 0; // Standard primitives virtual void rect( wxPoint p1, wxPoint p2, FILL_T fill, int width = -1 ) = 0; virtual void circle( wxPoint pos, int diametre, FILL_T fill, int width = -1 ) = 0; virtual void arc( wxPoint centre, int StAngle, int EndAngle, int rayon, FILL_T fill, int width = -1 ); /** * Function PlotPoly * @brief Draw a polygon ( filled or not ) * @param aCornerList = corners list * @param aFill :if true : filled polygon * @param aWidth = line width */ virtual void PlotPoly( std::vector< wxPoint >& aCornerList, FILL_T aFill, int aWidth = -1 ) = 0; /** * Function PlotImage * Only Postscript plotters can plot bitmaps * for plotters that cannot plot a bitmap, a rectangle is plotted * @brief Draw an image bitmap * @param aImage = the bitmap * @param aPos = position of the center of the bitmap * @param aScaleFactor = the scale factor to apply to the bitmap size * (this is not the plot scale factor) */ virtual void PlotImage( wxImage & aImage, wxPoint aPos, double aScaleFactor ) = 0; virtual void thick_segment( wxPoint start, wxPoint end, int width, EDA_DRAW_MODE_T tracemode ); virtual void thick_arc( wxPoint centre, int StAngle, int EndAngle, int rayon, int width, EDA_DRAW_MODE_T tracemode ); virtual void thick_rect( wxPoint p1, wxPoint p2, int width, EDA_DRAW_MODE_T tracemode ); virtual void thick_circle( wxPoint pos, int diametre, int width, EDA_DRAW_MODE_T tracemode ); virtual void pen_to( wxPoint pos, char plume ) = 0; // Flash primitives virtual void flash_pad_circle( wxPoint pos, int diametre, EDA_DRAW_MODE_T trace_mode ) = 0; virtual void flash_pad_oval( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ) = 0; virtual void flash_pad_rect( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ) = 0; /** virtual function flash_pad_trapez * flash a trapezoidal pad * @param aPadPos = the position of the shape * @param aCorners = the list of 4 corners positions, relative to the shape position, pad orientation 0 * @param aPadOrient = the rotation of the shape * @param aTrace_Mode = FILLED or SKETCH */ virtual void flash_pad_trapez( wxPoint aPadPos, wxPoint aCorners[4], int aPadOrient, EDA_DRAW_MODE_T aTrace_Mode ) = 0; // Convenience functions void move_to( wxPoint pos ) { pen_to( pos, 'U' ); } void line_to( wxPoint pos ) { pen_to( pos, 'D' ); } void finish_to( wxPoint pos ) { pen_to( pos, 'D' ); pen_to( pos, 'Z' ); } void pen_finish() { // Shortcut pen_to( wxPoint( 0, 0 ), 'Z' ); } void text( const wxPoint& aPos, enum EDA_Colors aColor, const wxString& aText, int aOrient, const wxSize& aSize, enum EDA_TEXT_HJUSTIFY_T aH_justify, enum EDA_TEXT_VJUSTIFY_T aV_justify, int aWidth, bool aItalic, bool aBold ); void marker( const wxPoint& position, int diametre, int aShapeId ); /** * Function SetLayerPolarity * sets current Gerber layer polarity to positive or negative * by writing \%LPD*\% or \%LPC*\% to the Gerber file, respectively. * @param aPositive is the layer polarity and true for positive. */ virtual void SetLayerPolarity( bool aPositive ) = 0; protected: // These are marker subcomponents void center_square( const wxPoint& position, int diametre, FILL_T fill ); void center_lozenge( const wxPoint& position, int diametre, FILL_T fill ); // Helper function for sketched filler segment void segment_as_oval( wxPoint start, wxPoint end, int width, EDA_DRAW_MODE_T tracemode ); void sketch_oval( wxPoint pos, wxSize size, int orient, int width ); virtual void user_to_device_coordinates( wxPoint& pos ); virtual void user_to_device_size( wxSize& size ); virtual double user_to_device_size( double size ); PlotFormat m_PlotType; /// Plot scale double plot_scale; /// Device scale (from decimils to device units) double device_scale; /// Plot offset (in decimils) wxPoint plot_offset; /// Output file FILE* output_file; // Pen handling bool color_mode, negative_mode; int default_pen_width; int current_pen_width; char pen_state; wxPoint pen_lastpos; bool plotMirror; wxString creator; wxString filename; PAGE_INFO pageInfo; wxSize paper_size; }; class HPGL_PLOTTER : public PLOTTER { public: HPGL_PLOTTER() : PLOTTER( PLOT_FORMAT_HPGL ) { } virtual bool start_plot( FILE* fout ); virtual bool end_plot(); // HPGL doesn't handle line thickness or color virtual void set_current_line_width( int width ) { // Handy override current_pen_width = wxRound( pen_diameter ); } virtual void set_default_line_width( int width ) {}; virtual void set_dash( bool dashed ); virtual void set_color( int color ) {}; virtual void set_pen_speed( int speed ) { wxASSERT( output_file == 0 ); pen_speed = speed; } virtual void set_pen_number( int number ) { wxASSERT( output_file == 0 ); pen_number = number; } virtual void set_pen_diameter( double diameter ) { wxASSERT( output_file == 0 ); pen_diameter = diameter; } virtual void set_pen_overlap( double overlap ) { wxASSERT( output_file == 0 ); pen_overlap = overlap; } virtual void set_viewport( wxPoint aOffset, double aScale, bool aMirror ); virtual void rect( wxPoint p1, wxPoint p2, FILL_T fill, int width = -1 ); virtual void circle( wxPoint pos, int diametre, FILL_T fill, int width = -1 ); /* * Function PlotPoly * Draw a polygon (filled or not) in HPGL format * param aCornerList = corners list * param aFill :if true : filled polygon * param aWidth = line width */ virtual void PlotPoly( std::vector< wxPoint >& aCornerList, FILL_T aFill, int aWidth = -1); /* * Function PlotImage * Only Postscript plotters can plot bitmaps * for plotters that cannot plot a bitmap, a rectangle is plotted * Draw an image bitmap * param aImage = the bitmap * param aPos = position of the center of the bitmap * param aScaleFactor = the scale factor to apply to the bitmap size * (this is not the plot scale factor) */ virtual void PlotImage( wxImage & aImage, wxPoint aPos, double aScaleFactor ); virtual void thick_segment( wxPoint start, wxPoint end, int width, EDA_DRAW_MODE_T tracemode ); virtual void arc( wxPoint centre, int StAngle, int EndAngle, int rayon, FILL_T fill, int width = -1 ); virtual void pen_to( wxPoint pos, char plume ); virtual void flash_pad_circle( wxPoint pos, int diametre, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_oval( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_rect( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_trapez( wxPoint aPadPos, wxPoint aCorners[4], int aPadOrient, EDA_DRAW_MODE_T aTrace_Mode ); virtual void SetLayerPolarity( bool aPositive ) {} protected: void pen_control( int plume ); int pen_speed; int pen_number; double pen_diameter; double pen_overlap; }; class PS_PLOTTER : public PLOTTER { public: PS_PLOTTER() : PLOTTER( PLOT_FORMAT_POST ) { plot_scale_adjX = 1; plot_scale_adjY = 1; } virtual bool start_plot( FILE* fout ); virtual bool end_plot(); virtual void set_current_line_width( int width ); virtual void set_default_line_width( int width ); virtual void set_dash( bool dashed ); virtual void set_color( int color ); void set_scale_adjust( double scaleX, double scaleY ) { plot_scale_adjX = scaleX; plot_scale_adjY = scaleY; } virtual void set_plot_width_adj( double width ) { plot_width_adj = width; } virtual double get_plot_width_adj() { return plot_width_adj; } virtual void set_viewport( wxPoint aOffset, double aScale, bool aMirror ); virtual void rect( wxPoint p1, wxPoint p2, FILL_T fill, int width = -1 ); virtual void circle( wxPoint pos, int diametre, FILL_T fill, int width = -1 ); virtual void arc( wxPoint centre, int StAngle, int EndAngle, int rayon, FILL_T fill, int width = -1 ); /* * Function PlotPoly * Draw a polygon (filled or not) in POSTSCRIPT format * param aCornerList = corners list * param aFill :if true : filled polygon * param aWidth = line width */ virtual void PlotPoly( std::vector< wxPoint >& aCornerList, FILL_T aFill, int aWidth = -1); /* * Function PlotImage * Only Postscript plotters can plot bitmaps * for plotters that cannot plot a bitmap, a rectangle is plotted * Draw an image bitmap * param aImage = the bitmap * param aPos = position of the center of the bitmap * param aScaleFactor = the scale factor to apply to the bitmap size * (this is not the plot scale factor) */ virtual void PlotImage( wxImage & aImage, wxPoint aPos, double aScaleFactor ); virtual void pen_to( wxPoint pos, char plume ); virtual void flash_pad_circle( wxPoint pos, int diametre, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_oval( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_rect( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_trapez( wxPoint aPadPos, wxPoint aCorners[4], int aPadOrient, EDA_DRAW_MODE_T aTrace_Mode ); virtual void SetLayerPolarity( bool aPositive ) {} protected: double plot_scale_adjX, plot_scale_adjY; double plot_width_adj; }; /* Class to handle a D_CODE when plotting a board : */ #define FIRST_DCODE_VALUE 10 // D_CODE < 10 is a command, D_CODE >= 10 is a tool struct APERTURE { enum Aperture_Type { Circle = 1, Rect = 2, Plotting = 3, Oval = 4 }; wxSize size; // horiz and Vert size Aperture_Type type; // Type ( Line, rect , circulaire , ovale .. ) int D_code; // code number ( >= 10 ); /* Trivia question: WHY Gerber decided to use D instead of the usual T for * tool change? */ }; class GERBER_PLOTTER : public PLOTTER { public: GERBER_PLOTTER() : PLOTTER( PLOT_FORMAT_GERBER ) { work_file = 0; final_file = 0; current_aperture = apertures.end(); } virtual bool start_plot( FILE* fout ); virtual bool end_plot(); virtual void set_current_line_width( int width ); virtual void set_default_line_width( int width ); // RS274X has no dashing, nor colours virtual void set_dash( bool dashed ) {}; virtual void set_color( int color ) {}; virtual void set_viewport( wxPoint aOffset, double aScale, bool aMirror ); virtual void rect( wxPoint p1, wxPoint p2, FILL_T fill, int width = -1 ); virtual void circle( wxPoint pos, int diametre, FILL_T fill, int width = -1 ); /* * Function PlotPoly * Draw a polygon (filled or not) in GERBER format * param aCornerList = corners list * param aFill :if true : filled polygon * param aWidth = line width */ virtual void PlotPoly( std::vector< wxPoint >& aCornerList, FILL_T aFill, int aWidth = -1); /* * Function PlotImage * Only Postscript plotters can plot bitmaps * for plotters that cannot plot a bitmap, a rectangle is plotted * Draw an image bitmap * param aImage = the bitmap * param aPos = position of the center of the bitmap * param aScaleFactor = the scale factor to apply to the bitmap size * (this is not the plot scale factor) */ virtual void PlotImage( wxImage & aImage, wxPoint aPos, double aScaleFactor ); virtual void pen_to( wxPoint pos, char plume ); virtual void flash_pad_circle( wxPoint pos, int diametre, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_oval( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_rect( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_trapez( wxPoint aPadPos, wxPoint aCorners[4], int aPadOrient, EDA_DRAW_MODE_T aTrace_Mode ); virtual void SetLayerPolarity( bool aPositive ); protected: void select_aperture( const wxSize& size, APERTURE::Aperture_Type type ); std::vector::iterator get_aperture( const wxSize& size, APERTURE::Aperture_Type type ); FILE* work_file, * final_file; wxString m_workFilename; void write_aperture_list(); std::vector apertures; std::vector::iterator current_aperture; }; class DXF_PLOTTER : public PLOTTER { public: DXF_PLOTTER() : PLOTTER( PLOT_FORMAT_DXF ) { } virtual bool start_plot( FILE* fout ); virtual bool end_plot(); // For now we don't use 'thick' primitives, so no line width virtual void set_current_line_width( int width ) { // Handy override current_pen_width = 0; } virtual void set_default_line_width( int width ) { // DXF lines are infinitesimal default_pen_width = 0; } virtual void set_dash( bool dashed ); virtual void set_color( int color ); virtual void set_viewport( wxPoint aOffset, double aScale, bool aMirror ); virtual void rect( wxPoint p1, wxPoint p2, FILL_T fill, int width = -1 ); virtual void circle( wxPoint pos, int diametre, FILL_T fill, int width = -1 ); /* * Function PlotPoly * Draw a polygon (filled or not) in DXF format * param aCornerList = corners list * param aFill :if true : filled polygon * param aWidth = line width */ virtual void PlotPoly( std::vector< wxPoint >& aCornerList, FILL_T aFill, int aWidth = -1 ); /* * Function PlotImage * Only Postscript plotters can plot bitmaps * for plotters that cannot plot a bitmap, a rectangle is plotted * Draw an image bitmap * param aImage = the bitmap * param aPos = position of the center of the bitmap * param aScaleFactor = the scale factor to apply to the bitmap size * (this is not the plot scale factor) */ virtual void PlotImage( wxImage & aImage, wxPoint aPos, double aScaleFactor ); virtual void thick_segment( wxPoint start, wxPoint end, int width, EDA_DRAW_MODE_T tracemode ); virtual void arc( wxPoint centre, int StAngle, int EndAngle, int rayon, FILL_T fill, int width = -1 ); virtual void pen_to( wxPoint pos, char plume ); virtual void flash_pad_circle( wxPoint pos, int diametre, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_oval( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_rect( wxPoint pos, wxSize size, int orient, EDA_DRAW_MODE_T trace_mode ); virtual void flash_pad_trapez( wxPoint aPadPos, wxPoint aCorners[4], int aPadOrient, EDA_DRAW_MODE_T aTrace_Mode ); virtual void SetLayerPolarity( bool aPositive ) {} protected: int current_color; }; #endif // PLOT_COMMON_H_