You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							354 lines
						
					
					
						
							10 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							354 lines
						
					
					
						
							10 KiB
						
					
					
				| /* | |
|  * This program source code file is part of KiCad, a free EDA CAD application. | |
|  * | |
|  * Copyright (C) 2018 Jean-Pierre Charras, jp.charras at wanadoo.fr | |
|  * Copyright (C) 2004-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 2 | |
|  * 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, you may find one here: | |
|  * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html | |
|  * or you may search the http://www.gnu.org website for the version 2 license, | |
|  * or you may write to the Free Software Foundation, Inc., | |
|  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA | |
|  */ | |
| 
 | |
| /** | |
|  * @file eda_rect.h | |
|  */ | |
| 
 | |
| #ifndef EDA_RECT_H | |
| #define EDA_RECT_H | |
|  | |
| #include <wx/gdicmn.h> | |
| #include <math/box2.h> | |
|  | |
| /** | |
|  * Handle the component boundary box. | |
|  * | |
|  * This class is similar to wxRect, but some wxRect functions are very curious, and are | |
|  * working only if dimensions are >= 0 (not always the case in KiCad) and also KiCad needs | |
|  * some specific method which makes this a more suitable class. | |
|  */ | |
| class EDA_RECT | |
| { | |
| public: | |
|     EDA_RECT() : m_init( false ) { }; | |
| 
 | |
|     EDA_RECT( const wxPoint& aPos, const wxSize& aSize ) : | |
|             m_pos( aPos ), | |
|             m_size( aSize ), | |
|             m_init( true ) | |
|     { } | |
| 
 | |
|     virtual ~EDA_RECT() { }; | |
| 
 | |
|     wxPoint Centre() const | |
|     { | |
|         return wxPoint( m_pos.x + ( m_size.x >> 1 ), m_pos.y + ( m_size.y >> 1 ) ); | |
|     } | |
| 
 | |
|     /** | |
|      * Move the rectangle by the \a aMoveVector. | |
|      * | |
|      * @param aMoveVector A wxPoint that is the value to move this rectangle. | |
|      */ | |
|     void Move( const wxPoint& aMoveVector ); | |
| 
 | |
|     /** | |
|      * Ensures that the height ant width are positive. | |
|      */ | |
|     void Normalize(); | |
| 
 | |
|     /** | |
|      * @param aPoint the wxPoint to test. | |
|      * @return true if aPoint is inside the boundary box. A point on a edge is seen as inside. | |
|      */ | |
|     bool Contains( const wxPoint& aPoint ) const; | |
| 
 | |
|     /** | |
|      * @param x the x coordinate of the point to test. | |
|      * @param y the x coordinate of the point to test. | |
|      * @return true if point is inside the boundary box. A point on a edge is seen as inside | |
|      */ | |
|     bool Contains( int x, int y ) const { return Contains( wxPoint( x, y ) ); } | |
| 
 | |
|     /** | |
|      * @param aRect the EDA_RECT to test. | |
|      * @return true if aRect is Contained. A common edge is seen as contained. | |
|      */ | |
|     bool Contains( const EDA_RECT& aRect ) const; | |
| 
 | |
|     const wxSize GetSize() const { return m_size; } | |
| 
 | |
|     /** | |
|      * @return the max size dimension. | |
|      */ | |
|     int GetSizeMax() const { return ( m_size.x > m_size.y ) ? m_size.x : m_size.y; } | |
| 
 | |
|     int GetX() const { return m_pos.x; } | |
|     int GetY() const { return m_pos.y; } | |
| 
 | |
|     const wxPoint GetOrigin() const { return m_pos; } | |
|     const wxPoint GetPosition() const { return m_pos; } | |
|     const wxPoint GetEnd() const { return wxPoint( m_pos.x + m_size.x, m_pos.y + m_size.y ); } | |
|     const wxPoint GetCenter() const | |
|     { | |
|         return wxPoint( m_pos.x + ( m_size.x / 2 ), m_pos.y + ( m_size.y / 2 ) ); | |
|     } | |
| 
 | |
|     int GetWidth() const { return m_size.x; } | |
|     int GetHeight() const { return m_size.y; } | |
|     int GetRight() const { return m_pos.x + m_size.x; } | |
|     int GetLeft() const { return m_pos.x; } | |
|     int GetTop() const { return m_pos.y; } | |
|     int GetBottom() const { return m_pos.y + m_size.y; }    // Y axis from top to bottom | |
|  | |
|     bool IsValid() const | |
|     { | |
|         return m_init; | |
|     } | |
| 
 | |
|     void SetOrigin( const wxPoint& pos ) | |
|     { | |
|         m_pos = pos; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetOrigin( int x, int y ) | |
|     { | |
|         m_pos.x = x; | |
|         m_pos.y = y; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetSize( const wxSize& size ) | |
|     { | |
|         m_size = size; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetSize( int w, int h ) | |
|     { | |
|         m_size.x = w; | |
|         m_size.y = h; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void Offset( int dx, int dy ) | |
|     { | |
|         m_pos.x += dx; | |
|         m_pos.y += dy; | |
|     } | |
| 
 | |
|     void Offset( const wxPoint& offset ) | |
|     { | |
|         m_pos += offset; | |
|     } | |
| 
 | |
|     void SetX( int val ) | |
|     { | |
|         m_pos.x = val; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetY( int val ) | |
|     { | |
|         m_pos.y = val; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetWidth( int val ) | |
|     { | |
|         m_size.x = val; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetHeight( int val ) | |
|     { | |
|         m_size.y = val; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetEnd( int x, int y ) | |
|     { | |
|         SetEnd( wxPoint( x, y ) ); | |
|         m_init = true; | |
|     } | |
| 
 | |
|     void SetEnd( const wxPoint& pos ) | |
|     { | |
|         m_size.x = pos.x - m_pos.x; | |
|         m_size.y = pos.y - m_pos.y; | |
|         m_init = true; | |
|     } | |
| 
 | |
|     /** | |
|      * Mirror the rectangle from the X axis (negate Y pos and size). | |
|      */ | |
|     void RevertYAxis() | |
|     { | |
|         m_pos.y  = -m_pos.y; | |
|         m_size.y = -m_size.y; | |
|         Normalize(); | |
|     } | |
| 
 | |
|     /** | |
|      * Test for a common area between rectangles. | |
|      * | |
|      * @param aRect A rectangle to test intersection with. | |
|      * @return true if the argument rectangle intersects this rectangle. | |
|      * (i.e. if the 2 rectangles have at least a common point) | |
|      */ | |
|     bool Intersects( const EDA_RECT& aRect ) const; | |
| 
 | |
|     /** | |
|      * Tests for a common area between this rectangle, and a rectangle with arbitrary rotation | |
|      * | |
|      * @param aRect a rectangle to test intersection with. | |
|      * @param aRot rectangle rotation (in 1/10 degrees). | |
|      */ | |
|     bool Intersects( const EDA_RECT& aRect, double aRot ) const; | |
| 
 | |
|     /** | |
|      * Test for a common area between a segment and this rectangle. | |
|      * | |
|      * @param aPoint1 First point of the segment to test intersection with. | |
|      * @param aPoint2 Second point of the segment to test intersection with. | |
|      * @return true if the argument segment intersects this rectangle. | |
|      * (i.e. if the segment and rectangle have at least a common point) | |
|      */ | |
|     bool Intersects( const wxPoint& aPoint1, const wxPoint& aPoint2 ) const; | |
| 
 | |
|     /** | |
|      * Test for intersection between a segment and this rectangle, returning the intersections. | |
|      * | |
|      * @param aPoint1 is the first point of the segment to test intersection with. | |
|      * @param aPoint2 is the second point of the segment to test intersection with. | |
|      * @param aIntersection1 will be filled with the first intersection point, if any. | |
|      * @param aIntersection2 will be filled with the second intersection point, if any. | |
|      * @return true if the segment intersects the rect. | |
|      */ | |
|     bool Intersects( const wxPoint& aPoint1, const wxPoint& aPoint2, | |
|                      wxPoint* aIntersection1, wxPoint* aIntersection2 ) const; | |
| 
 | |
|     /** | |
|      * Return the point in this rect that is closest to the provided point | |
|      */ | |
|     const wxPoint ClosestPointTo( const wxPoint& aPoint ) const; | |
| 
 | |
|     /** | |
|      * Return the point in this rect that is farthest from the provided point | |
|      */ | |
|     const wxPoint FarthestPointTo( const wxPoint& aPoint ) const; | |
| 
 | |
|     /** | |
|      * Test for a common area between a circle and this rectangle. | |
|      * | |
|      * @param aCenter center of the circle. | |
|      * @param aRadius radius of the circle. | |
|      */ | |
|     bool IntersectsCircle( const wxPoint& aCenter, const int aRadius ) const; | |
| 
 | |
|     /** | |
|      * Test for intersection between this rect and the edge (radius) of a circle. | |
|      * | |
|      * @param aCenter center of the circle. | |
|      * @param aRadius radius of the circle. | |
|      * @param aWidth width of the circle edge. | |
|      */ | |
|     bool IntersectsCircleEdge( const wxPoint& aCenter, const int aRadius, const int aWidth ) const; | |
| 
 | |
|     /** | |
|      * Overload the cast operator to return a wxRect. | |
|      * | |
|      * wxRect does not accept negative values for size, so ensure the wxRect size is always >= 0. | |
|      */ | |
|     operator wxRect() const | |
|     { | |
|         EDA_RECT rect( m_pos, m_size ); | |
|         rect.Normalize(); | |
|         return wxRect( rect.m_pos, rect.m_size ); | |
|     } | |
| 
 | |
|     /** | |
|      * Overload the cast operator to return a BOX2I. | |
|      * | |
|      * @return this box shaped as a BOX2I object. | |
|      */ | |
|     operator BOX2I() const | |
|     { | |
|         EDA_RECT rect( m_pos, m_size ); | |
|         rect.Normalize(); | |
|         return BOX2I( rect.GetOrigin(), rect.GetSize() ); | |
|     } | |
| 
 | |
|     /** | |
|      * Inflate the rectangle horizontally by \a dx and vertically by \a dy. If \a dx | |
|      * and/or \a dy is negative the rectangle is deflated. | |
|      */ | |
|     EDA_RECT& Inflate( wxCoord dx, wxCoord dy ); | |
| 
 | |
|     /** | |
|      * Inflate the rectangle horizontally and vertically by \a aDelta. If \a aDelta | |
|      * is negative the rectangle is deflated. | |
|      */ | |
|     EDA_RECT& Inflate( int aDelta ); | |
| 
 | |
|     /** | |
|      * Modify the position and size of the rectangle in order to contain \a aRect. | |
|      * | |
|      * It is mainly used to calculate bounding boxes. | |
|      * | |
|      * @param aRect  The rectangle to merge with this rectangle. | |
|      */ | |
|     void Merge( const EDA_RECT& aRect ); | |
| 
 | |
|     /** | |
|      * Modify the position and size of the rectangle in order to contain the given point. | |
|      * | |
|      * @param aPoint The point to merge with the rectangle. | |
|      */ | |
|     void Merge( const wxPoint& aPoint ); | |
| 
 | |
|     /** | |
|      * Return the area of the rectangle. | |
|      * | |
|      * @return The area of the rectangle. | |
|      */ | |
|     double GetArea() const; | |
| 
 | |
|     /** | |
|      * Return the area that is common with another rectangle. | |
|      * | |
|      * @param aRect is the rectangle to find the common area with. | |
|      * @return The common area rect or 0-sized rectangle if there is no intersection. | |
|      */ | |
|     EDA_RECT Common( const EDA_RECT& aRect ) const; | |
| 
 | |
|     /** | |
|      * Useful to calculate bounding box of rotated items, when rotation if not k*90 degrees. | |
|      * | |
|      * @param aAngle the rotation angle in 0.1 deg. | |
|      * @param aRotCenter the rotation point. | |
|      * @return the bounding box of this, after rotation. | |
|      */ | |
|     const EDA_RECT GetBoundingBoxRotated( const wxPoint& aRotCenter, double aAngle ) const; | |
| 
 | |
| private: | |
|     wxPoint m_pos;      // Rectangle Origin | |
|     wxSize  m_size;     // Rectangle Size | |
|     bool    m_init;     // Is the rectangle initialized | |
| }; | |
| 
 | |
| 
 | |
| #endif // EDA_RECT_H
 |