2009-08-29 10:20:48 +00:00
|
|
|
/******************************************
|
|
|
|
* class_plotter.cpp
|
|
|
|
* the class PLOTTER handle basic functions to plot schematic and boards
|
|
|
|
* with different plot formats.
|
|
|
|
* currently formats are:*
|
|
|
|
* HPGL
|
|
|
|
* POSTSCRIPT
|
|
|
|
* GERBER
|
|
|
|
* DXF
|
|
|
|
******************************************/
|
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <fctsys.h>
|
2009-08-29 10:20:48 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <trigo.h>
|
|
|
|
#include <wxstruct.h>
|
|
|
|
#include <base_struct.h>
|
|
|
|
#include <common.h>
|
|
|
|
#include <plot_common.h>
|
|
|
|
#include <worksheet.h>
|
|
|
|
#include <macros.h>
|
|
|
|
#include <class_base_screen.h>
|
|
|
|
#include <drawtxt.h>
|
2009-08-29 10:20:48 +00:00
|
|
|
|
|
|
|
PLOTTER::PLOTTER( PlotFormat aPlotType )
|
|
|
|
{
|
|
|
|
m_PlotType = aPlotType;
|
|
|
|
plot_scale = 1;
|
|
|
|
default_pen_width = 0;
|
|
|
|
current_pen_width = -1; /* To-be-set marker */
|
|
|
|
pen_state = 'Z'; /* End-of-path idle */
|
2010-12-11 18:40:39 +00:00
|
|
|
plotMirror = 0; /* Mirror flag */
|
2009-08-29 10:20:48 +00:00
|
|
|
output_file = 0;
|
|
|
|
color_mode = false; /* Start as a BW plot */
|
|
|
|
negative_mode = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
/* Modifies coordinates pos.x and pos.y trace according to the orientation,
|
|
|
|
* scale factor, and offsets trace
|
|
|
|
*/
|
2009-08-29 10:20:48 +00:00
|
|
|
void PLOTTER::user_to_device_coordinates( wxPoint& pos )
|
|
|
|
{
|
|
|
|
pos.x = (int) ( (pos.x - plot_offset.x) * plot_scale * device_scale );
|
|
|
|
|
2010-12-11 18:40:39 +00:00
|
|
|
if( plotMirror )
|
2009-11-23 15:16:50 +00:00
|
|
|
pos.y = (int) ( ( pos.y - plot_offset.y ) * plot_scale * device_scale );
|
2009-08-29 10:20:48 +00:00
|
|
|
else
|
2009-11-23 15:16:50 +00:00
|
|
|
pos.y = (int) ( ( paper_size.y - ( pos.y - plot_offset.y )
|
|
|
|
* plot_scale ) * device_scale );
|
2009-08-29 10:20:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Generic arc rendered as a polyline */
|
2009-11-23 15:16:50 +00:00
|
|
|
void PLOTTER::arc( wxPoint centre, int StAngle, int EndAngle, int radius,
|
|
|
|
FILL_T fill, int width )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
|
|
|
wxPoint start, end;
|
|
|
|
const int delta = 50; /* increment (in 0.1 degrees) to draw circles */
|
|
|
|
double alpha;
|
|
|
|
|
|
|
|
if( StAngle > EndAngle )
|
|
|
|
EXCHG( StAngle, EndAngle );
|
|
|
|
|
|
|
|
set_current_line_width( width );
|
|
|
|
/* Please NOTE the different sign due to Y-axis flip */
|
|
|
|
alpha = StAngle / 1800.0 * M_PI;
|
2009-11-23 15:16:50 +00:00
|
|
|
start.x = centre.x + (int) ( radius * cos( -alpha ) );
|
|
|
|
start.y = centre.y + (int) ( radius * sin( -alpha ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
move_to( start );
|
|
|
|
for( int ii = StAngle + delta; ii < EndAngle; ii += delta )
|
|
|
|
{
|
|
|
|
alpha = ii / 1800.0 * M_PI;
|
2009-11-23 15:16:50 +00:00
|
|
|
end.x = centre.x + (int) ( radius * cos( -alpha ) );
|
|
|
|
end.y = centre.y + (int) ( radius * sin( -alpha ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
line_to( end );
|
|
|
|
}
|
|
|
|
|
|
|
|
alpha = EndAngle / 1800.0 * M_PI;
|
2009-11-23 15:16:50 +00:00
|
|
|
end.x = centre.x + (int) ( radius * cos( -alpha ) );
|
|
|
|
end.y = centre.y + (int) ( radius * sin( -alpha ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
finish_to( end );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
/* Modifies size size.x and size.y trace according to the scale factor. */
|
2009-08-29 10:20:48 +00:00
|
|
|
void PLOTTER::user_to_device_size( wxSize& size )
|
|
|
|
{
|
|
|
|
size.x = (int) ( size.x * plot_scale * device_scale );
|
|
|
|
size.y = (int) ( size.y * plot_scale * device_scale );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
double PLOTTER::user_to_device_size( double size )
|
|
|
|
{
|
|
|
|
return size * plot_scale * device_scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PLOTTER::center_square( const wxPoint& position, int diametre, FILL_T fill )
|
|
|
|
{
|
// Dick Hollenbeck's KiROUND R&D
// This provides better project control over rounding to int from double
// than wxRound() did. This scheme provides better logging in Debug builds
// and it provides for compile time calculation of constants.
#include <stdio.h>
#include <assert.h>
#include <limits.h>
//-----<KiROUND KIT>------------------------------------------------------------
/**
* KiROUND
* rounds a floating point number to an int using
* "round halfway cases away from zero".
* In Debug build an assert fires if will not fit into an int.
*/
#if defined( DEBUG )
// DEBUG: a macro to capture line and file, then calls this inline
static inline int KiRound( double v, int line, const char* filename )
{
v = v < 0 ? v - 0.5 : v + 0.5;
if( v > INT_MAX + 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' > 0 ' for int\n", __FUNCTION__, filename, line, v );
}
else if( v < INT_MIN - 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' < 0 ' for int\n", __FUNCTION__, filename, line, v );
}
return int( v );
}
#define KiROUND( v ) KiRound( v, __LINE__, __FILE__ )
#else
// RELEASE: a macro so compile can pre-compute constants.
#define KiROUND( v ) int( (v) < 0 ? (v) - 0.5 : (v) + 0.5 )
#endif
//-----</KiROUND KIT>-----------------------------------------------------------
// Only a macro is compile time calculated, an inline function causes a static constructor
// in a situation like this.
// Therefore the Release build is best done with a MACRO not an inline function.
int Computed = KiROUND( 14.3 * 8 );
int main( int argc, char** argv )
{
for( double d = double(INT_MAX)-1; d < double(INT_MAX)+8; d += 2.0 )
{
int i = KiROUND( d );
printf( "t: %d %.16g\n", i, d );
}
return 0;
}
2012-04-19 06:55:45 +00:00
|
|
|
int radius = KiROUND( diametre / 2.8284 );
|
2011-04-20 08:13:21 +00:00
|
|
|
static std::vector< wxPoint > corner_list;
|
|
|
|
corner_list.clear();
|
|
|
|
wxPoint corner;
|
|
|
|
corner.x = position.x + radius;
|
|
|
|
corner.y = position.y + radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x + radius;
|
|
|
|
corner.y = position.y - radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x - radius;
|
|
|
|
corner.y = position.y - radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x - radius;
|
|
|
|
corner.y = position.y + radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x + radius;
|
|
|
|
corner.y = position.y + radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
|
|
|
|
PlotPoly( corner_list, fill );
|
2009-08-29 10:20:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-20 08:13:21 +00:00
|
|
|
void PLOTTER::center_lozenge( const wxPoint& position, int diametre, FILL_T fill )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
2009-11-23 15:16:50 +00:00
|
|
|
int radius = diametre / 2;
|
2011-04-20 08:13:21 +00:00
|
|
|
static std::vector< wxPoint > corner_list;
|
|
|
|
corner_list.clear();
|
|
|
|
wxPoint corner;
|
|
|
|
corner.x = position.x;
|
|
|
|
corner.y = position.y + radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x + radius;
|
|
|
|
corner.y = position.y,
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x;
|
|
|
|
corner.y = position.y - radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x - radius;
|
|
|
|
corner.y = position.y;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
corner.x = position.x;
|
|
|
|
corner.y = position.y + radius;
|
|
|
|
corner_list.push_back( corner );
|
|
|
|
|
|
|
|
PlotPoly( corner_list, fill );
|
2009-08-29 10:20:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
/* Draw a pattern shape number aShapeId, to coord x0, y0.
|
|
|
|
* x0, y0 = coordinates tables
|
|
|
|
* Diameter diameter = (coord table) hole
|
|
|
|
* AShapeId = index (used to generate forms characters)
|
2009-08-29 10:20:48 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
void PLOTTER::marker( const wxPoint& position, int diametre, int aShapeId )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
2009-11-23 15:16:50 +00:00
|
|
|
int radius = diametre / 2;
|
2009-08-29 10:20:48 +00:00
|
|
|
|
|
|
|
int x0, y0;
|
|
|
|
|
|
|
|
x0 = position.x; y0 = position.y;
|
|
|
|
|
|
|
|
switch( aShapeId )
|
|
|
|
{
|
2009-11-23 15:16:50 +00:00
|
|
|
case 0: /* vias : X shape */
|
|
|
|
move_to( wxPoint( x0 - radius, y0 - radius ) );
|
|
|
|
line_to( wxPoint( x0 + radius, y0 + radius ) );
|
|
|
|
move_to( wxPoint( x0 + radius, y0 - radius ) );
|
|
|
|
finish_to( wxPoint( x0 - radius, y0 + radius ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 1: /* Circle */
|
2009-08-29 10:20:48 +00:00
|
|
|
circle( position, diametre, NO_FILL );
|
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 2: /* + shape */
|
|
|
|
move_to( wxPoint( x0, y0 - radius ) );
|
|
|
|
line_to( wxPoint( x0, y0 + radius ) );
|
|
|
|
move_to( wxPoint( x0 + radius, y0 ) );
|
|
|
|
finish_to( wxPoint( x0 - radius, y0 ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 3: /* X shape in circle */
|
2009-08-29 10:20:48 +00:00
|
|
|
circle( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0 - radius, y0 - radius ) );
|
|
|
|
line_to( wxPoint( x0 + radius, y0 + radius ) );
|
|
|
|
move_to( wxPoint( x0 + radius, y0 - radius ) );
|
|
|
|
finish_to( wxPoint( x0 - radius, y0 + radius ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 4: /* circle with bar - shape */
|
2009-08-29 10:20:48 +00:00
|
|
|
circle( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0 - radius, y0 ) );
|
|
|
|
finish_to( wxPoint( x0 + radius, y0 ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 5: /* circle with bar | shape */
|
2009-08-29 10:20:48 +00:00
|
|
|
circle( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0, y0 - radius ) );
|
|
|
|
finish_to( wxPoint( x0, y0 + radius ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 6: /* square */
|
2009-08-29 10:20:48 +00:00
|
|
|
center_square( position, diametre, NO_FILL );
|
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 7: /* diamond */
|
2009-08-29 10:20:48 +00:00
|
|
|
center_lozenge( position, diametre, NO_FILL );
|
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 8: /* square with an X*/
|
2009-08-29 10:20:48 +00:00
|
|
|
center_square( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0 - radius, y0 - radius ) );
|
|
|
|
line_to( wxPoint( x0 + radius, y0 + radius ) );
|
|
|
|
move_to( wxPoint( x0 + radius, y0 - radius ) );
|
|
|
|
finish_to( wxPoint( x0 - radius, y0 + radius ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 9: /* diamond with a +*/
|
2009-08-29 10:20:48 +00:00
|
|
|
center_lozenge( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0, y0 - radius ) );
|
|
|
|
line_to( wxPoint( x0, y0 + radius ) );
|
|
|
|
move_to( wxPoint( x0 + radius, y0 ) );
|
|
|
|
finish_to( wxPoint( x0 - radius, y0 ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 10: /* square with a '/' */
|
2009-08-29 10:20:48 +00:00
|
|
|
center_square( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0 - radius, y0 - radius ) );
|
|
|
|
finish_to( wxPoint( x0 + radius, y0 + radius ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 11: /* square with a |*/
|
2009-08-29 10:20:48 +00:00
|
|
|
center_lozenge( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0, y0 - radius ) );
|
|
|
|
finish_to( wxPoint( x0, y0 + radius ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
case 12: /* square with a -*/
|
2009-08-29 10:20:48 +00:00
|
|
|
center_lozenge( position, diametre, NO_FILL );
|
2009-11-23 15:16:50 +00:00
|
|
|
move_to( wxPoint( x0 - radius, y0 ) );
|
|
|
|
finish_to( wxPoint( x0 + radius, y0 ) );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
circle( position, diametre, NO_FILL );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
/* Convert a thick segment and plot it as an oval */
|
2009-08-29 10:20:48 +00:00
|
|
|
void PLOTTER::segment_as_oval( wxPoint start, wxPoint end, int width,
|
2012-01-03 17:14:17 +00:00
|
|
|
EDA_DRAW_MODE_T tracemode )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
|
|
|
wxPoint center( (start.x + end.x) / 2, (start.y + end.y) / 2 );
|
|
|
|
wxSize size( end.x - start.x, end.y - start.y );
|
|
|
|
int orient;
|
|
|
|
|
|
|
|
if( size.y == 0 )
|
|
|
|
orient = 0;
|
|
|
|
else if( size.x == 0 )
|
|
|
|
orient = 900;
|
|
|
|
else
|
2009-11-23 15:16:50 +00:00
|
|
|
orient = -(int) ( atan2( (double) size.y,
|
|
|
|
(double) size.x ) * 1800.0 / M_PI );
|
|
|
|
size.x = (int) sqrt( ( (double) size.x * size.x )
|
|
|
|
+ ( (double) size.y * size.y ) ) + width;
|
2009-08-29 10:20:48 +00:00
|
|
|
size.y = width;
|
|
|
|
|
|
|
|
flash_pad_oval( center, size, orient, tracemode );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PLOTTER::sketch_oval( wxPoint pos, wxSize size, int orient,
|
|
|
|
int width )
|
|
|
|
{
|
|
|
|
set_current_line_width( width );
|
|
|
|
width = current_pen_width;
|
2009-11-23 15:16:50 +00:00
|
|
|
int radius, deltaxy, cx, cy;
|
|
|
|
|
2009-08-29 10:20:48 +00:00
|
|
|
if( size.x > size.y )
|
|
|
|
{
|
2009-11-23 15:16:50 +00:00
|
|
|
EXCHG( size.x, size.y );
|
|
|
|
orient += 900;
|
2009-08-29 10:20:48 +00:00
|
|
|
if( orient >= 3600 )
|
|
|
|
orient -= 3600;
|
|
|
|
}
|
2009-11-23 15:16:50 +00:00
|
|
|
|
|
|
|
deltaxy = size.y - size.x; /* distance between centers of the oval */
|
|
|
|
radius = ( size.x - width ) / 2;
|
|
|
|
cx = -radius;
|
|
|
|
cy = -deltaxy / 2;
|
2009-08-29 10:20:48 +00:00
|
|
|
RotatePoint( &cx, &cy, orient );
|
|
|
|
move_to( wxPoint( cx + pos.x, cy + pos.y ) );
|
2009-11-23 15:16:50 +00:00
|
|
|
cx = -radius;
|
|
|
|
cy = deltaxy / 2;
|
2009-08-29 10:20:48 +00:00
|
|
|
RotatePoint( &cx, &cy, orient );
|
|
|
|
finish_to( wxPoint( cx + pos.x, cy + pos.y ) );
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
cx = radius;
|
|
|
|
cy = -deltaxy / 2;
|
2009-08-29 10:20:48 +00:00
|
|
|
RotatePoint( &cx, &cy, orient );
|
|
|
|
move_to( wxPoint( cx + pos.x, cy + pos.y ) );
|
2009-11-23 15:16:50 +00:00
|
|
|
cx = radius;
|
|
|
|
cy = deltaxy / 2;
|
2009-08-29 10:20:48 +00:00
|
|
|
RotatePoint( &cx, &cy, orient );
|
|
|
|
finish_to( wxPoint( cx + pos.x, cy + pos.y ) );
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
cx = 0;
|
|
|
|
cy = deltaxy / 2;
|
2009-08-29 10:20:48 +00:00
|
|
|
RotatePoint( &cx, &cy, orient );
|
|
|
|
arc( wxPoint( cx + pos.x, cy + pos.y ),
|
|
|
|
orient + 1800, orient + 3600,
|
2009-11-23 15:16:50 +00:00
|
|
|
radius, NO_FILL );
|
|
|
|
cx = 0;
|
|
|
|
cy = -deltaxy / 2;
|
2009-08-29 10:20:48 +00:00
|
|
|
RotatePoint( &cx, &cy, orient );
|
|
|
|
arc( wxPoint( cx + pos.x, cy + pos.y ),
|
|
|
|
orient, orient + 1800,
|
2009-11-23 15:16:50 +00:00
|
|
|
radius, NO_FILL );
|
2009-08-29 10:20:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Plot 1 segment like a track segment
|
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
void PLOTTER::thick_segment( wxPoint start, wxPoint end, int width,
|
2012-01-03 17:14:17 +00:00
|
|
|
EDA_DRAW_MODE_T tracemode )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
|
|
|
switch( tracemode )
|
|
|
|
{
|
|
|
|
case FILLED:
|
2012-01-03 17:14:17 +00:00
|
|
|
case LINE:
|
2009-08-29 10:20:48 +00:00
|
|
|
set_current_line_width( tracemode==FILLED ? width : -1 );
|
|
|
|
move_to( start );
|
|
|
|
finish_to( end );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SKETCH:
|
|
|
|
set_current_line_width( -1 );
|
|
|
|
segment_as_oval( start, end, width, tracemode );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
void PLOTTER::thick_arc( wxPoint centre, int StAngle, int EndAngle, int radius,
|
2012-01-03 17:14:17 +00:00
|
|
|
int width, EDA_DRAW_MODE_T tracemode )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
|
|
|
switch( tracemode )
|
|
|
|
{
|
2012-01-03 17:14:17 +00:00
|
|
|
case LINE:
|
2009-08-29 10:20:48 +00:00
|
|
|
set_current_line_width( -1 );
|
2009-11-23 15:16:50 +00:00
|
|
|
arc( centre, StAngle, EndAngle, radius, NO_FILL, -1 );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case FILLED:
|
2009-11-23 15:16:50 +00:00
|
|
|
arc( centre, StAngle, EndAngle, radius, NO_FILL, width );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SKETCH:
|
|
|
|
set_current_line_width( -1 );
|
2009-11-23 15:16:50 +00:00
|
|
|
arc( centre, StAngle, EndAngle,
|
|
|
|
radius - ( width - current_pen_width ) / 2, NO_FILL, -1 );
|
|
|
|
arc( centre, StAngle, EndAngle,
|
|
|
|
radius + ( width - current_pen_width ) / 2, NO_FILL, -1 );
|
2009-08-29 10:20:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PLOTTER::thick_rect( wxPoint p1, wxPoint p2, int width,
|
2012-01-03 17:14:17 +00:00
|
|
|
EDA_DRAW_MODE_T tracemode )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
|
|
|
switch( tracemode )
|
|
|
|
{
|
2012-01-03 17:14:17 +00:00
|
|
|
case LINE:
|
2009-08-29 10:20:48 +00:00
|
|
|
rect( p1, p2, NO_FILL, -1 );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FILLED:
|
|
|
|
rect( p1, p2, NO_FILL, width );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SKETCH:
|
|
|
|
set_current_line_width( -1 );
|
|
|
|
p1.x -= (width - current_pen_width) / 2;
|
|
|
|
p1.y -= (width - current_pen_width) / 2;
|
|
|
|
p2.x += (width - current_pen_width) / 2;
|
|
|
|
p2.y += (width - current_pen_width) / 2;
|
|
|
|
rect( p1, p2, NO_FILL, -1 );
|
|
|
|
p1.x += (width - current_pen_width);
|
|
|
|
p1.y += (width - current_pen_width);
|
|
|
|
p2.x -= (width - current_pen_width);
|
|
|
|
p2.y -= (width - current_pen_width);
|
|
|
|
rect( p1, p2, NO_FILL, -1 );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PLOTTER::thick_circle( wxPoint pos, int diametre, int width,
|
2012-01-03 17:14:17 +00:00
|
|
|
EDA_DRAW_MODE_T tracemode )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
|
|
|
switch( tracemode )
|
|
|
|
{
|
2012-01-03 17:14:17 +00:00
|
|
|
case LINE:
|
2009-08-29 10:20:48 +00:00
|
|
|
circle( pos, diametre, NO_FILL, -1 );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FILLED:
|
|
|
|
circle( pos, diametre, NO_FILL, width );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SKETCH:
|
|
|
|
set_current_line_width( -1 );
|
|
|
|
circle( pos, diametre - width + current_pen_width, NO_FILL, -1 );
|
|
|
|
circle( pos, diametre + width - current_pen_width, NO_FILL, -1 );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-22 21:57:50 +00:00
|
|
|
void PLOTTER::SetPageSettings( const PAGE_INFO& aPageSettings )
|
2009-08-29 10:20:48 +00:00
|
|
|
{
|
|
|
|
wxASSERT( !output_file );
|
2011-12-22 21:57:50 +00:00
|
|
|
pageInfo = aPageSettings;
|
2009-08-29 10:20:48 +00:00
|
|
|
|
2011-12-31 05:44:00 +00:00
|
|
|
// PAGE_INFO is in mils, plotter works with deci-mils
|
2011-12-22 21:57:50 +00:00
|
|
|
paper_size = pageInfo.GetSizeMils() * 10;
|
2009-08-29 10:20:48 +00:00
|
|
|
}
|
2011-12-22 21:57:50 +00:00
|
|
|
|