kicad/common/gr_basic.cpp

827 lines
22 KiB
C++
Raw Normal View History

/*
* This program source code file is part of KiCad, a free EDA CAD application.
*
* Copyright (C) 2015 Jean-Pierre Charras, jp.charras at wanadoo.fr
* Copyright (C) 2011 Wayne Stambaugh <stambaughw@gmail.com>
* Copyright (C) 1992-2021 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 as published by the
* Free Software Foundation, either version 3 of the License, or (at your
* option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <gr_basic.h>
#include <trigo.h>
#include <eda_item.h>
#include <math_for_graphics.h>
#include <wx/graphics.h>
2018-09-07 21:33:01 +00:00
#include <geometry/geometry_utils.h>
#include <math/util.h> // for KiROUND
2009-06-25 20:45:27 +00:00
2020-07-11 09:37:22 +00:00
#include <algorithm>
static const bool FILLED = true;
static const bool NOT_FILLED = false;
/* Important Note:
* These drawing functions clip draw item before send these items to wxDC draw
* functions. For guy who asks why i did it, see a sample of problems encountered
* when pixels
* coordinates overflow 16 bits values:
* http://trac.wxwidgets.org/ticket/10446
* Problems can be found under Windows **and** Linux (mainly when drawing arcs)
* (mainly at low zoom values (2, 1 or 0.5), in Pcbnew)
* some of these problems could be now fixed in recent distributions.
*
* Currently (feb 2009) there are overflow problems when drawing solid (filled)
* polygons under linux without clipping
*
* So before removing clipping functions, be aware these bug (they are not in
* KiCad or wxWidgets) are fixed by testing how are drawn complex lines arcs
* and solid polygons under Windows and Linux and remember users can have old
* versions with bugs
*/
/* Definitions for enabling and disabling debugging features in gr_basic.cpp.
* Please remember to set these back to 0 before making LAUNCHPAD commits.
*/
#define DEBUG_DUMP_CLIP_ERROR_COORDS 0 // Set to 1 to dump clip algorithm errors.
#define DEBUG_DUMP_CLIP_COORDS 0 // Set to 1 to dump clipped coordinates.
// For draw mode = XOR GR_XOR or GR_NXOR by background color
GR_DRAWMODE g_XorMode = GR_NXOR;
static void ClipAndDrawPoly( EDA_RECT* ClipBox, wxDC* DC, const VECTOR2I* Points, int n );
/* These functions are used by corresponding functions
* ( GRSCircle is called by GRCircle for instance) after mapping coordinates
* from user units to screen units(pixels coordinates)
*/
static void GRSRect( EDA_RECT* aClipBox, wxDC* aDC, int x1, int y1,
2021-11-17 10:59:49 +00:00
int x2, int y2, int aWidth, const COLOR4D& aColor );
/**/
2021-11-17 10:59:49 +00:00
static int GRLastMoveToX, GRLastMoveToY;
static bool s_ForceBlackPen; /* if true: draws in black instead of
* color for printing. */
2021-11-17 10:59:49 +00:00
static int xcliplo = 0,
ycliplo = 0,
xcliphi = 2000,
ycliphi = 2000;
2010-09-21 19:21:46 +00:00
2021-11-17 10:59:49 +00:00
static COLOR4D s_DC_lastbrushcolor( 0, 0, 0, 0 );
static bool s_DC_lastbrushfill = false;
static wxDC* s_DC_lastDC = nullptr;
2007-10-31 06:40:15 +00:00
static void WinClipAndDrawLine( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2,
int width )
2007-10-31 06:40:15 +00:00
{
GRLastMoveToX = x2;
GRLastMoveToY = y2;
if( ClipBox )
{
EDA_RECT clipbox(*ClipBox);
clipbox.Inflate(width/2);
2018-09-07 21:33:01 +00:00
if( ClipLine( &clipbox, x1, y1, x2, y2 ) )
2007-10-31 17:44:51 +00:00
return;
2007-10-31 06:40:15 +00:00
}
DC->DrawLine( x1, y1, x2, y2 );
}
2007-10-31 06:40:15 +00:00
void GRResetPenAndBrush( wxDC* DC )
{
2007-10-31 06:40:15 +00:00
GRSetBrush( DC, BLACK ); // Force no fill
s_DC_lastbrushcolor = COLOR4D::UNSPECIFIED;
s_DC_lastDC = nullptr;
}
2007-10-31 06:40:15 +00:00
void GRSetColorPen( wxDC* DC, const COLOR4D& Color, int width, wxPenStyle style )
{
COLOR4D color = Color;
wxDash dots[2] = { 1, 3 };
2015-03-25 09:55:00 +00:00
// Under OSX and while printing when wxPen is set to 0, renderer follows the request drawing
// nothing & in the bitmap world the minimum is enough to light a pixel, in vectorial one not
if( width <= 1 && DC->GetBrush().GetStyle() != wxBRUSHSTYLE_SOLID )
width = DC->DeviceToLogicalXRel( 1 );
if( s_ForceBlackPen )
color = COLOR4D::BLACK;
const wxPen& curr_pen = DC->GetPen();
2008-04-02 14:16:14 +00:00
if( !curr_pen.IsOk() || curr_pen.GetColour() != color.ToColour()
|| curr_pen.GetWidth() != width || curr_pen.GetStyle() != style )
{
wxPen pen;
pen.SetColour( color.ToColour() );
if( style == wxPENSTYLE_DOT )
{
style = wxPENSTYLE_USER_DASH;
pen.SetDashes( 2, dots );
}
2008-04-02 14:16:14 +00:00
pen.SetWidth( width );
pen.SetStyle( style );
DC->SetPen( pen );
2007-10-31 06:40:15 +00:00
}
else
{
// Should be not needed, but on Linux, in printing process
// the curr pen settings needs to be sometimes re-initialized
// Clearly, this is due to a bug, related to SetBrush(),
// but we have to live with it, at least on wxWidgets 3.0
DC->SetPen( curr_pen );
}
}
2007-10-31 06:40:15 +00:00
void GRSetBrush( wxDC* DC, const COLOR4D& Color, bool fill )
{
COLOR4D color = Color;
if( s_ForceBlackPen )
color = COLOR4D::BLACK;
if( s_DC_lastbrushcolor != color || s_DC_lastbrushfill != fill || s_DC_lastDC != DC )
2010-09-21 19:21:46 +00:00
{
wxBrush brush;
brush.SetColour( color.ToColour() );
2010-09-21 19:21:46 +00:00
if( fill )
brush.SetStyle( wxBRUSHSTYLE_SOLID );
else
brush.SetStyle( wxBRUSHSTYLE_TRANSPARENT );
2010-09-21 19:21:46 +00:00
DC->SetBrush( brush );
2010-09-21 19:21:46 +00:00
s_DC_lastbrushcolor = color;
s_DC_lastbrushfill = fill;
s_DC_lastDC = DC;
2010-09-21 19:21:46 +00:00
}
}
2007-10-31 06:40:15 +00:00
void GRForceBlackPen( bool flagforce )
{
s_ForceBlackPen = flagforce;
}
bool GetGRForceBlackPenState( void )
{
return s_ForceBlackPen;
}
2007-10-31 06:40:15 +00:00
void GRLine( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2, int width,
const COLOR4D& Color, wxPenStyle aStyle)
{
GRSetColorPen( DC, Color, width, aStyle );
WinClipAndDrawLine( ClipBox, DC, x1, y1, x2, y2, width );
GRLastMoveToX = x2;
GRLastMoveToY = y2;
2009-05-25 16:07:33 +00:00
}
2007-10-31 06:40:15 +00:00
void GRLine( EDA_RECT* aClipBox, wxDC* aDC, const VECTOR2I& aStart, const VECTOR2I& aEnd,
int aWidth, const COLOR4D& aColor, wxPenStyle aStyle )
{
GRLine( aClipBox, aDC, aStart.x, aStart.y, aEnd.x, aEnd.y, aWidth, aColor, aStyle );
}
2007-10-31 06:40:15 +00:00
void GRMoveTo( int x, int y )
{
GRLastMoveToX = x;
GRLastMoveToY = y;
}
2007-10-31 06:40:15 +00:00
void GRLineTo( EDA_RECT* ClipBox, wxDC* DC, int x, int y, int width, const COLOR4D& Color )
{
GRLine( ClipBox, DC, GRLastMoveToX, GRLastMoveToY, x, y, width, Color );
}
2007-10-31 06:40:15 +00:00
void GRCSegm( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2, int width,
int aPenSize, const COLOR4D& Color )
2007-10-31 06:40:15 +00:00
{
GRLastMoveToX = x2;
GRLastMoveToY = y2;
if( ClipBox )
{
EDA_RECT clipbox( *ClipBox );
clipbox.Inflate( width / 2 );
2007-10-31 06:40:15 +00:00
2018-09-07 21:33:01 +00:00
if( ClipLine( &clipbox, x1, y1, x2, y2 ) )
2007-10-31 17:44:51 +00:00
return;
2007-10-31 06:40:15 +00:00
}
if( width <= 2 ) /* single line or 2 pixels */
2007-10-31 06:40:15 +00:00
{
GRSetColorPen( DC, Color, width );
DC->DrawLine( x1, y1, x2, y2 );
return;
}
2015-03-25 17:27:17 +00:00
GRSetBrush( DC, Color, NOT_FILLED );
GRSetColorPen( DC, Color, aPenSize );
2007-10-31 06:40:15 +00:00
int radius = ( width + 1 ) >> 1;
int dx = x2 - x1;
int dy = y2 - y1;
2022-01-14 15:34:41 +00:00
EDA_ANGLE angle( VECTOR2I( dx, dy ) );
angle = -angle;
2022-01-05 02:23:11 +00:00
VECTOR2I start;
VECTOR2I end;
VECTOR2I org( x1, y1 );
int len = (int) hypot( dx, dy );
// We know if the DC is mirrored, to draw arcs
int slx = DC->DeviceToLogicalX( 1 ) - DC->DeviceToLogicalX( 0 );
int sly = DC->DeviceToLogicalY( 1 ) - DC->DeviceToLogicalY( 0 );
bool mirrored = ( slx > 0 && sly < 0 ) || ( slx < 0 && sly > 0 );
// first edge
start.x = 0;
start.y = radius;
end.x = len;
end.y = radius;
2022-01-05 02:23:11 +00:00
RotatePoint( start, angle );
RotatePoint( end, angle );
start += org;
end += org;
2022-01-05 02:23:11 +00:00
DC->DrawLine( (wxPoint) start, (wxPoint) end );
// first rounded end
end.x = 0;
end.y = -radius;
2022-01-05 02:23:11 +00:00
RotatePoint( end, angle );
end += org;
if( !mirrored )
2022-01-05 02:23:11 +00:00
DC->DrawArc( (wxPoint) end, (wxPoint) start, (wxPoint) org );
2007-10-31 06:40:15 +00:00
else
2022-01-05 02:23:11 +00:00
DC->DrawArc( (wxPoint) start, (wxPoint) end, (wxPoint) org );
2008-04-02 14:16:14 +00:00
// second edge
start.x = len;
start.y = -radius;
2022-01-05 02:23:11 +00:00
RotatePoint( start, angle );
start += org;
2007-10-31 06:40:15 +00:00
2022-01-05 02:23:11 +00:00
DC->DrawLine( (wxPoint) start, (wxPoint) end );
2008-04-02 14:16:14 +00:00
// second rounded end
end.x = len;
end.y = radius;
2022-01-05 02:23:11 +00:00
RotatePoint( end, angle);
end += org;
2007-10-31 06:40:15 +00:00
if( !mirrored )
DC->DrawArc( end.x, end.y, start.x, start.y, x2, y2 );
2007-10-31 06:40:15 +00:00
else
DC->DrawArc( start.x, start.y, end.x, end.y, x2, y2 );
2007-10-31 06:40:15 +00:00
}
void GRCSegm( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2, int width,
const COLOR4D& Color )
{
GRCSegm( ClipBox, DC, x1, y1, x2, y2, width, 0, Color );
}
void GRCSegm( EDA_RECT* aClipBox, wxDC* aDC, const VECTOR2I& aStart, const VECTOR2I& aEnd,
int aWidth, const COLOR4D& aColor )
{
GRCSegm( aClipBox, aDC, aStart.x, aStart.y, aEnd.x, aEnd.y, aWidth, 0, aColor );
}
2021-11-17 10:59:49 +00:00
void GRFillCSegm( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2, int width,
const COLOR4D& Color )
{
GRSetColorPen( DC, Color, width );
WinClipAndDrawLine( ClipBox, DC, x1, y1, x2, y2, width );
}
void GRFilledSegment( EDA_RECT* aClipBox, wxDC* aDC, const VECTOR2I& aStart, const VECTOR2I& aEnd,
int aWidth, const COLOR4D& aColor )
{
GRSetColorPen( aDC, aColor, aWidth );
WinClipAndDrawLine( aClipBox, aDC, aStart.x, aStart.y, aEnd.x, aEnd.y, aWidth );
}
static bool IsGRSPolyDrawable( EDA_RECT* ClipBox, int n, const VECTOR2I* Points )
2007-10-31 06:40:15 +00:00
{
if( !ClipBox )
return true;
if( n <= 0 )
return false;
2007-10-31 06:40:15 +00:00
int Xmin, Xmax, Ymin, Ymax;
Xmin = Xmax = Points[0].x;
Ymin = Ymax = Points[0].y;
2007-10-31 06:40:15 +00:00
for( int ii = 1; ii < n; ii++ ) // calculate rectangle
2007-10-31 06:40:15 +00:00
{
Xmin = std::min( Xmin, Points[ii].x );
Xmax = std::max( Xmax, Points[ii].x );
Ymin = std::min( Ymin, Points[ii].y );
Ymax = std::max( Ymax, Points[ii].y );
2007-10-31 06:40:15 +00:00
}
xcliplo = ClipBox->GetX();
ycliplo = ClipBox->GetY();
xcliphi = ClipBox->GetRight();
ycliphi = ClipBox->GetBottom();
2007-10-31 06:40:15 +00:00
if( Xmax < xcliplo )
return false;
2007-10-31 06:40:15 +00:00
if( Xmin > xcliphi )
return false;
2007-10-31 06:40:15 +00:00
if( Ymax < ycliplo )
return false;
2007-10-31 06:40:15 +00:00
if( Ymin > ycliphi )
return false;
2007-10-31 06:40:15 +00:00
return true;
}
2007-10-31 06:40:15 +00:00
/**
* Draw a new polyline and fill it if Fill, in screen space.
*/
static void GRSPoly( EDA_RECT* ClipBox, wxDC* DC, int n, const VECTOR2I* Points, bool Fill,
int width, const COLOR4D& Color, const COLOR4D& BgColor )
{
2007-10-31 06:40:15 +00:00
if( !IsGRSPolyDrawable( ClipBox, n, Points ) )
return;
2007-10-31 06:40:15 +00:00
if( Fill && ( n > 2 ) )
{
GRSetBrush( DC, BgColor, FILLED );
2015-03-25 17:27:17 +00:00
GRSetColorPen( DC, Color, width );
/* clip before send the filled polygon to wxDC, because under linux
* (GTK?) polygons having large coordinates are incorrectly drawn
* (integer overflow in coordinates, I am guessing)
*/
ClipAndDrawPoly( ClipBox, DC, Points, n );
2007-10-31 06:40:15 +00:00
}
else
{
GRMoveTo( Points[0].x, Points[0].y );
for( int i = 1; i < n; ++i )
{
GRLineTo( ClipBox, DC, Points[i].x, Points[i].y, width, Color );
}
2007-10-31 06:40:15 +00:00
}
}
/**
* Draw a new closed polyline and fill it if Fill, in screen space.
*/
static void GRSClosedPoly( EDA_RECT* aClipBox, wxDC* aDC, int aPointCount, const VECTOR2I* aPoints,
bool aFill, int aWidth, const COLOR4D& aColor, const COLOR4D& aBgColor )
{
if( !IsGRSPolyDrawable( aClipBox, aPointCount, aPoints ) )
2007-10-31 06:40:15 +00:00
return;
if( aFill && ( aPointCount > 2 ) )
2007-10-31 06:40:15 +00:00
{
GRLastMoveToX = aPoints[aPointCount - 1].x;
GRLastMoveToY = aPoints[aPointCount - 1].y;
GRSetBrush( aDC, aBgColor, FILLED );
2015-03-25 17:27:17 +00:00
GRSetColorPen( aDC, aColor, aWidth );
ClipAndDrawPoly( aClipBox, aDC, aPoints, aPointCount );
2007-10-31 06:40:15 +00:00
}
else
{
GRMoveTo( aPoints[0].x, aPoints[0].y );
for( int i = 1; i < aPointCount; ++i )
{
GRLineTo( aClipBox, aDC, aPoints[i].x, aPoints[i].y, aWidth, aColor );
}
int lastpt = aPointCount - 1;
// Close the polygon
if( aPoints[lastpt] != aPoints[0] )
2007-10-31 06:40:15 +00:00
{
GRLineTo( aClipBox, aDC, aPoints[0].x, aPoints[0].y, aWidth, aColor );
2007-10-31 06:40:15 +00:00
}
}
}
/**
* Draw a new polyline and fill it if Fill, in drawing space.
*/
void GRPoly( EDA_RECT* ClipBox, wxDC* DC, int n, const VECTOR2I* Points, bool Fill, int width,
const COLOR4D& Color, const COLOR4D& BgColor )
{
2007-10-31 06:40:15 +00:00
GRSPoly( ClipBox, DC, n, Points, Fill, width, Color, BgColor );
}
2007-10-31 06:40:15 +00:00
/**
* Draw a closed polyline and fill it if Fill, in object space.
*/
void GRClosedPoly( EDA_RECT* ClipBox, wxDC* DC, int n, const VECTOR2I* Points, bool Fill,
const COLOR4D& Color, const COLOR4D& BgColor )
{
2007-10-31 06:40:15 +00:00
GRClosedPoly( ClipBox, DC, n, Points, Fill, 0, Color, BgColor );
}
2007-10-31 06:40:15 +00:00
void GRClosedPoly( EDA_RECT* ClipBox, wxDC* DC, int n, const VECTOR2I* Points, bool Fill, int width,
const COLOR4D& Color, const COLOR4D& BgColor )
{
2007-10-31 06:40:15 +00:00
GRSClosedPoly( ClipBox, DC, n, Points, Fill, width, Color, BgColor );
}
2007-10-31 06:40:15 +00:00
2017-10-26 08:55:01 +00:00
static bool clipCircle( EDA_RECT* aClipBox, int xc, int yc, int r, int aWidth )
{
2017-10-26 08:55:01 +00:00
// Clip circles that are outside the ClipBox.
if( aClipBox )
{
int x0, y0, xm, ym;
2017-10-26 08:55:01 +00:00
x0 = aClipBox->GetX();
y0 = aClipBox->GetY();
xm = aClipBox->GetRight();
ym = aClipBox->GetBottom();
r += aWidth;
2017-10-26 08:55:01 +00:00
if( xc < ( x0 - r ) )
2017-10-26 08:19:15 +00:00
return true;
2017-10-26 08:55:01 +00:00
if( yc < ( y0 - r ) )
2017-10-26 08:19:15 +00:00
return true;
2017-10-26 08:55:01 +00:00
if( xc > ( r + xm ) )
2017-10-26 08:19:15 +00:00
return true;
2010-09-28 14:42:05 +00:00
2017-10-26 08:55:01 +00:00
if( yc > ( r + ym ) )
2017-10-26 08:19:15 +00:00
return true;
}
2017-10-26 08:19:15 +00:00
return false;
}
void GRCircle( EDA_RECT* ClipBox, wxDC* DC, int xc, int yc, int r, int width, const COLOR4D& Color )
2017-10-26 08:19:15 +00:00
{
if( clipCircle( ClipBox, xc, yc, r, width ) || r <= 0 )
2017-10-26 08:19:15 +00:00
return;
2015-03-25 17:27:17 +00:00
GRSetBrush( DC, Color, NOT_FILLED );
GRSetColorPen( DC, Color, width );
// Draw two arcs here to make a circle. Unfortunately, the printerDC doesn't handle
// transparent brushes when used with circles. It does work for for arcs, however
DC->DrawArc(xc + r, yc, xc - r, yc, xc, yc);
DC->DrawArc(xc - r, yc, xc + r, yc, xc, yc);
2010-09-28 14:42:05 +00:00
}
2007-10-31 06:40:15 +00:00
void GRCircle( EDA_RECT* aClipBox, wxDC* aDC, const VECTOR2I& aPos, int aRadius, int aWidth,
const COLOR4D& aColor )
2010-09-28 14:42:05 +00:00
{
GRCircle( aClipBox, aDC, aPos.x, aPos.y, aRadius, aWidth, aColor );
2010-09-28 14:42:05 +00:00
}
2007-10-31 06:40:15 +00:00
void GRFilledCircle( EDA_RECT* ClipBox, wxDC* DC, int x, int y, int r, int width,
const COLOR4D& Color, const COLOR4D& BgColor )
2007-10-31 06:40:15 +00:00
{
if( clipCircle( ClipBox, x, y, r, width ) || r <= 0 )
2017-10-26 08:19:15 +00:00
return;
2007-10-31 06:40:15 +00:00
GRSetBrush( DC, BgColor, FILLED );
2015-03-25 17:27:17 +00:00
GRSetColorPen( DC, Color, width );
2007-10-31 06:40:15 +00:00
DC->DrawEllipse( x - r, y - r, r + r, r + r );
}
void GRFilledCircle( EDA_RECT* aClipBox, wxDC* aDC, const VECTOR2I& aPos, int aRadius,
const COLOR4D& aColor )
{
GRFilledCircle( aClipBox, aDC, aPos.x, aPos.y, aRadius, 0, aColor, aColor );
}
void GRArc1( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2, int xc, int yc,
int width, const COLOR4D& Color )
2007-10-31 06:40:15 +00:00
{
/* Clip arcs off screen. */
2007-10-31 06:40:15 +00:00
if( ClipBox )
{
int x0, y0, xm, ym, r;
x0 = ClipBox->GetX();
y0 = ClipBox->GetY();
xm = ClipBox->GetRight();
ym = ClipBox->GetBottom();
r = KiROUND( Distance( x1, y1, xc, yc ) );
if( xc < ( x0 - r ) )
2007-10-31 06:40:15 +00:00
return;
if( yc < ( y0 - r ) )
2007-10-31 06:40:15 +00:00
return;
if( xc > ( r + xm ) )
2007-10-31 06:40:15 +00:00
return;
if( yc > ( r + ym ) )
2007-10-31 06:40:15 +00:00
return;
}
GRSetBrush( DC, Color );
2015-03-25 17:27:17 +00:00
GRSetColorPen( DC, Color, width );
2007-10-31 06:40:15 +00:00
DC->DrawArc( x1, y1, x2, y2, xc, yc );
}
2007-10-31 06:40:15 +00:00
void GRArc1( EDA_RECT* aClipBox, wxDC* aDC, const VECTOR2I& aStart, const VECTOR2I& aEnd,
const VECTOR2I& aCenter, int aWidth, const COLOR4D& aColor )
{
GRArc1( aClipBox, aDC, aStart.x, aStart.y, aEnd.x, aEnd.y, aCenter.x, aCenter.y,
aWidth, aColor );
}
2007-10-31 06:40:15 +00:00
void GRFilledArc1( EDA_RECT* ClipBox, wxDC* DC, const VECTOR2I& aStart, const VECTOR2I& aEnd,
const VECTOR2I& aCenter, int width, const COLOR4D& Color,
const COLOR4D& BgColor )
2021-11-17 01:39:16 +00:00
{
/* Clip arcs off screen. */
if( ClipBox )
{
int x0, y0, xm, ym, r;
x0 = ClipBox->GetX();
y0 = ClipBox->GetY();
xm = ClipBox->GetRight();
ym = ClipBox->GetBottom();
r = KiROUND( Distance( aStart.x, aStart.y, aCenter.x, aCenter.y ) );
if( aCenter.x < ( x0 - r ) )
return;
if( aCenter.y < ( y0 - r ) )
return;
if( aCenter.x > ( r + xm ) )
return;
if( aCenter.y > ( r + ym ) )
return;
}
GRSetBrush( DC, BgColor, FILLED );
GRSetColorPen( DC, Color, width );
DC->DrawArc( aStart.x, aStart.y, aEnd.x, aEnd.y, aCenter.x, aCenter.y );
}
void GRArc( EDA_RECT* ClipBox, wxDC* DC, int xc, int yc, double StAngle, double EndAngle, int r,
const COLOR4D& Color )
2007-10-31 06:40:15 +00:00
{
int x1, y1, x2, y2;
/* Clip arcs off screen */
2007-10-31 06:40:15 +00:00
if( ClipBox )
{
int radius = r + 1;
2007-10-31 06:40:15 +00:00
int x0, y0, xm, ym, x, y;
x0 = ClipBox->GetX();
y0 = ClipBox->GetY();
xm = ClipBox->GetRight();
ym = ClipBox->GetBottom();
x = xc;
y = yc;
if( x < ( x0 - radius ) )
2007-10-31 06:40:15 +00:00
return;
if( y < ( y0 - radius ) )
2007-10-31 06:40:15 +00:00
return;
if( x > ( xm + radius ) )
2007-10-31 06:40:15 +00:00
return;
if( y > ( ym + radius ) )
2007-10-31 06:40:15 +00:00
return;
}
x1 = r;
y1 = 0;
2007-10-31 06:40:15 +00:00
RotatePoint( &x1, &y1, EndAngle );
x2 = r;
y2 = 0;
2007-10-31 06:40:15 +00:00
RotatePoint( &x2, &y2, StAngle );
2015-03-25 17:27:17 +00:00
GRSetBrush( DC, Color, NOT_FILLED );
2007-10-31 06:40:15 +00:00
GRSetColorPen( DC, Color );
DC->DrawArc( xc + x1, yc - y1, xc + x2, yc - y2, xc, yc );
}
2007-10-31 06:40:15 +00:00
2021-11-17 10:59:49 +00:00
void GRArc( EDA_RECT* ClipBox, wxDC* DC, int x, int y, double StAngle, double EndAngle, int r,
int width, const COLOR4D& Color )
{
int x1, y1, x2, y2;
/* Clip arcs off screen. */
if( ClipBox )
{
int x0, y0, xm, ym;
x0 = ClipBox->GetX();
y0 = ClipBox->GetY();
xm = ClipBox->GetRight();
ym = ClipBox->GetBottom();
if( x < ( x0 - r - width ) )
return;
if( y < ( y0 - r - width ) )
return;
if( x > ( r + xm + width ) )
return;
if( y > ( r + ym + width ) )
return;
}
x1 = r;
y1 = 0;
RotatePoint( &x1, &y1, EndAngle );
x2 = r;
y2 = 0;
RotatePoint( &x2, &y2, StAngle );
GRSetBrush( DC, Color );
2015-03-25 17:27:17 +00:00
GRSetColorPen( DC, Color, width );
DC->DrawArc( x + x1, y - y1, x + x2, y - y2, x, y );
}
2007-10-31 06:40:15 +00:00
void GRRect( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2, int width,
const COLOR4D& Color )
{
2007-10-31 06:40:15 +00:00
GRSRect( ClipBox, DC, x1, y1, x2, y2, width, Color );
}
2007-10-31 06:40:15 +00:00
void GRFilledRect( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2,
const COLOR4D& Color, const COLOR4D& BgColor )
{
2007-10-31 06:40:15 +00:00
GRSFilledRect( ClipBox, DC, x1, y1, x2, y2, 0, Color, BgColor );
}
2007-10-31 06:40:15 +00:00
2021-11-17 10:59:49 +00:00
void GRFilledRect( EDA_RECT* ClipBox, wxDC* DC, int x1, int y1, int x2, int y2, int width,
const COLOR4D& Color, const COLOR4D& BgColor )
{
2007-10-31 06:40:15 +00:00
GRSFilledRect( ClipBox, DC, x1, y1, x2, y2, width, Color, BgColor );
}
2007-10-31 06:40:15 +00:00
2021-11-17 10:59:49 +00:00
void GRSRect( EDA_RECT* aClipBox, wxDC* aDC, int x1, int y1, int x2, int y2, int aWidth,
const COLOR4D& aColor )
{
VECTOR2I points[5];
points[0] = VECTOR2I( x1, y1 );
points[1] = VECTOR2I( x1, y2 );
points[2] = VECTOR2I( x2, y2 );
points[3] = VECTOR2I( x2, y1 );
points[4] = points[0];
GRSClosedPoly( aClipBox, aDC, 5, points, NOT_FILLED, aWidth, aColor, aColor );
}
2021-11-17 10:59:49 +00:00
void GRSFilledRect( EDA_RECT* aClipBox, wxDC* aDC, int x1, int y1, int x2, int y2, int aWidth,
const COLOR4D& aColor, const COLOR4D& aBgColor )
{
VECTOR2I points[5];
points[0] = VECTOR2I( x1, y1 );
points[1] = VECTOR2I( x1, y2 );
points[2] = VECTOR2I( x2, y2 );
points[3] = VECTOR2I( x2, y1 );
points[4] = points[0];
2015-03-25 17:27:17 +00:00
GRSetBrush( aDC, aBgColor, FILLED );
GRSetColorPen( aDC, aBgColor, aWidth );
if( aClipBox && ( aWidth > 0 ) )
{
EDA_RECT clipbox( *aClipBox );
clipbox.Inflate( aWidth );
ClipAndDrawPoly( &clipbox, aDC, points, 5 ); // polygon approach is more accurate
}
else
{
ClipAndDrawPoly(aClipBox, aDC, points, 5 );
}
2007-10-31 06:40:15 +00:00
}
/**
* Used to clip a polygon and draw it as Filled Polygon.
*
* Uses the Sutherland and Hodgman algo to clip the given poly against a rectangle. This
* rectangle is the drawing area this is useful under Linux (2009) because filled polygons
* are incorrectly drawn if they have too large coordinates (seems due to integer overflows
* in calculations). Could be removed in some years, if become unnecessary.
*/
2010-10-25 16:48:05 +00:00
#include <SutherlandHodgmanClipPoly.h>
static void vector2IwxDrawPolygon( wxDC* aDC, const VECTOR2I* Points, int n )
{
wxPoint* points = new wxPoint[n];
for( int i = 0; i < n; i++ )
{
points[i] = wxPoint( Points[i].x, Points[i].y );
}
aDC->DrawPolygon( n, points );
delete[] points;
}
void ClipAndDrawPoly( EDA_RECT* aClipBox, wxDC* aDC, const VECTOR2I* Points, int n )
{
if( aClipBox == nullptr )
2010-10-25 16:48:05 +00:00
{
vector2IwxDrawPolygon( aDC, Points, n );
2010-10-25 16:48:05 +00:00
return;
}
// A clip box exists: clip and draw the polygon.
static std::vector<VECTOR2I> clippedPolygon;
static pointVector inputPolygon, outputPolygon;
inputPolygon.clear();
outputPolygon.clear();
clippedPolygon.clear();
for( int ii = 0; ii < n; ii++ )
inputPolygon.push_back( PointF( (REAL) Points[ii].x, (REAL) Points[ii].y ) );
RectF window( (REAL) aClipBox->GetX(), (REAL) aClipBox->GetY(),
(REAL) aClipBox->GetWidth(), (REAL) aClipBox->GetHeight() );
SutherlandHodgman sh( window );
sh.Clip( inputPolygon, outputPolygon );
for( cpointIterator cit = outputPolygon.begin(); cit != outputPolygon.end(); ++cit )
{
clippedPolygon.emplace_back( KiROUND( cit->X ), KiROUND( cit->Y ) );
}
if( clippedPolygon.size() )
vector2IwxDrawPolygon( aDC, &clippedPolygon[0], clippedPolygon.size() );
}