summaryrefslogtreecommitdiff
path: root/pcbnew/class_dimension.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'pcbnew/class_dimension.cpp')
-rw-r--r--pcbnew/class_dimension.cpp523
1 files changed, 523 insertions, 0 deletions
diff --git a/pcbnew/class_dimension.cpp b/pcbnew/class_dimension.cpp
new file mode 100644
index 0000000..0367c14
--- /dev/null
+++ b/pcbnew/class_dimension.cpp
@@ -0,0 +1,523 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2012 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr
+ * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
+ * Copyright (C) 2012 Wayne Stambaugh <stambaughw@verizon.net>
+ * Copyright (C) 1992-2012 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 class_dimension.cpp
+ */
+
+#include <fctsys.h>
+#include <macros.h>
+#include <gr_basic.h>
+#include <trigo.h>
+#include <wxstruct.h>
+#include <class_drawpanel.h>
+#include <colors_selection.h>
+#include <kicad_string.h>
+#include <richio.h>
+
+#include <class_board.h>
+#include <class_pcb_text.h>
+#include <class_dimension.h>
+#include <base_units.h>
+
+
+DIMENSION::DIMENSION( BOARD_ITEM* aParent ) :
+ BOARD_ITEM( aParent, PCB_DIMENSION_T ),
+ m_Width( Millimeter2iu( 0.2 ) ), m_Unit( INCHES ), m_Value( 0 ), m_Height( 0 ), m_Text( this )
+{
+ m_Layer = Dwgs_User;
+ m_Shape = 0;
+}
+
+
+DIMENSION::~DIMENSION()
+{
+}
+
+
+void DIMENSION::SetPosition( const wxPoint& aPos )
+{
+ m_Text.SetTextPosition( aPos );
+}
+
+
+const wxPoint& DIMENSION::GetPosition() const
+{
+ return m_Text.GetTextPosition();
+}
+
+
+void DIMENSION::SetText( const wxString& aNewText )
+{
+ m_Text.SetText( aNewText );
+}
+
+
+const wxString DIMENSION::GetText() const
+{
+ return m_Text.GetText();
+}
+
+
+void DIMENSION::SetLayer( LAYER_ID aLayer )
+{
+ m_Layer = aLayer;
+ m_Text.SetLayer( aLayer );
+}
+
+
+void DIMENSION::Copy( DIMENSION* source )
+{
+ m_Value = source->m_Value;
+ SetLayer( source->GetLayer() );
+ m_Width = source->m_Width;
+ m_Shape = source->m_Shape;
+ m_Height = source->m_Height;
+ m_Unit = source->m_Unit;
+ SetTimeStamp( GetNewTimeStamp() );
+ m_Text.Copy( &source->m_Text );
+
+ m_crossBarO = source->m_crossBarO;
+ m_crossBarF = source->m_crossBarF;
+ m_featureLineGO = source->m_featureLineGO;
+ m_featureLineGF = source->m_featureLineGF;
+ m_featureLineDO = source->m_featureLineDO;
+ m_featureLineDF = source->m_featureLineDF;
+ m_arrowD1F = source->m_arrowD1F;
+ m_arrowD2F = source->m_arrowD2F;
+ m_arrowG1F = source->m_arrowG1F;
+ m_arrowG2F = source->m_arrowG2F;
+}
+
+
+void DIMENSION::Move( const wxPoint& offset )
+{
+ m_Text.SetTextPosition( m_Text.GetTextPosition() + offset );
+ m_crossBarO += offset;
+ m_crossBarF += offset;
+ m_featureLineGO += offset;
+ m_featureLineGF += offset;
+ m_featureLineDO += offset;
+ m_featureLineDF += offset;
+ m_arrowG1F += offset;
+ m_arrowG2F += offset;
+ m_arrowD1F += offset;
+ m_arrowD2F += offset;
+}
+
+
+void DIMENSION::Rotate( const wxPoint& aRotCentre, double aAngle )
+{
+ wxPoint tmp = m_Text.GetTextPosition();
+ RotatePoint( &tmp, aRotCentre, aAngle );
+ m_Text.SetTextPosition( tmp );
+
+ double newAngle = m_Text.GetOrientation() + aAngle;
+
+ if( newAngle >= 3600 )
+ newAngle -= 3600;
+
+ if( newAngle > 900 && newAngle < 2700 )
+ newAngle -= 1800;
+
+ m_Text.SetOrientation( newAngle );
+
+ RotatePoint( &m_crossBarO, aRotCentre, aAngle );
+ RotatePoint( &m_crossBarF, aRotCentre, aAngle );
+ RotatePoint( &m_featureLineGO, aRotCentre, aAngle );
+ RotatePoint( &m_featureLineGF, aRotCentre, aAngle );
+ RotatePoint( &m_featureLineDO, aRotCentre, aAngle );
+ RotatePoint( &m_featureLineDF, aRotCentre, aAngle );
+ RotatePoint( &m_arrowG1F, aRotCentre, aAngle );
+ RotatePoint( &m_arrowG2F, aRotCentre, aAngle );
+ RotatePoint( &m_arrowD1F, aRotCentre, aAngle );
+ RotatePoint( &m_arrowD2F, aRotCentre, aAngle );
+}
+
+
+void DIMENSION::Flip( const wxPoint& aCentre )
+{
+ Mirror( aCentre );
+
+ // DIMENSION items are not usually on copper layers, so
+ // copper layers count is not taken in accoun in Flip transform
+ SetLayer( FlipLayer( GetLayer() ) );
+}
+
+
+void DIMENSION::Mirror( const wxPoint& axis_pos )
+{
+ wxPoint newPos = m_Text.GetTextPosition();
+
+#define INVERT( pos ) (pos) = axis_pos.y - ( (pos) - axis_pos.y )
+ INVERT( newPos.y );
+
+ m_Text.SetTextPosition( newPos );
+
+ // invert angle
+ m_Text.SetOrientation( -m_Text.GetOrientation() );
+
+ INVERT( m_crossBarO.y );
+ INVERT( m_crossBarF.y );
+ INVERT( m_featureLineGO.y );
+ INVERT( m_featureLineGF.y );
+ INVERT( m_featureLineDO.y );
+ INVERT( m_featureLineDF.y );
+ INVERT( m_arrowG1F.y );
+ INVERT( m_arrowG2F.y );
+ INVERT( m_arrowD1F.y );
+ INVERT( m_arrowD2F.y );
+}
+
+
+void DIMENSION::SetOrigin( const wxPoint& aOrigin )
+{
+ m_featureLineGO = aOrigin;
+
+ AdjustDimensionDetails();
+}
+
+
+void DIMENSION::SetEnd( const wxPoint& aEnd )
+{
+ m_featureLineDO = aEnd;
+
+ AdjustDimensionDetails();
+}
+
+
+void DIMENSION::SetHeight( int aHeight )
+{
+ m_Height = aHeight;
+
+ AdjustDimensionDetails();
+}
+
+
+void DIMENSION::UpdateHeight()
+{
+ VECTOR2D featureLine( m_crossBarO - m_featureLineGO );
+ VECTOR2D crossBar( m_featureLineDO - m_featureLineGO );
+
+ if( featureLine.Cross( crossBar ) > 0 )
+ m_Height = -featureLine.EuclideanNorm();
+ else
+ m_Height = featureLine.EuclideanNorm();
+}
+
+
+void DIMENSION::AdjustDimensionDetails( bool aDoNotChangeText )
+{
+ const int arrowz = DMils2iu( 500 ); // size of arrows
+ int ii;
+ int measure, deltax, deltay; // value of the measure on X and Y axes
+ int arrow_up_X = 0, arrow_up_Y = 0; // coordinates of arrow line /
+ int arrow_dw_X = 0, arrow_dw_Y = 0; // coordinates of arrow line '\'
+ int hx, hy; // dimension line interval
+ double angle, angle_f;
+ wxString msg;
+
+ // Init layer :
+ m_Text.SetLayer( GetLayer() );
+
+ // calculate the size of the dimension (text + line above the text)
+ ii = m_Text.GetSize().y + m_Text.GetThickness() + (m_Width * 3);
+
+ deltax = m_featureLineDO.x - m_featureLineGO.x;
+ deltay = m_featureLineDO.y - m_featureLineGO.y;
+
+ // Calculate dimension value
+ measure = KiROUND( hypot( deltax, deltay ) );
+
+ angle = atan2( deltay, deltax );
+
+ // Calculation of parameters X and Y dimensions of the arrows and lines.
+ hx = hy = ii;
+
+ // Taking into account the slope of the side lines.
+ if( measure )
+ {
+ hx = abs( KiROUND( ( (double) deltay * hx ) / measure ) );
+ hy = abs( KiROUND( ( (double) deltax * hy ) / measure ) );
+
+ if( m_featureLineGO.x > m_crossBarO.x )
+ hx = -hx;
+
+ if( m_featureLineGO.x == m_crossBarO.x )
+ hx = 0;
+
+ if( m_featureLineGO.y > m_crossBarO.y )
+ hy = -hy;
+
+ if( m_featureLineGO.y == m_crossBarO.y )
+ hy = 0;
+
+ angle_f = angle + DEG2RAD( 27.5 );
+ arrow_up_X = wxRound( arrowz * cos( angle_f ) );
+ arrow_up_Y = wxRound( arrowz * sin( angle_f ) );
+ angle_f = angle - DEG2RAD( 27.5 );
+ arrow_dw_X = wxRound( arrowz * cos( angle_f ) );
+ arrow_dw_Y = wxRound( arrowz * sin( angle_f ) );
+ }
+
+ int dx = KiROUND( m_Height * cos( angle + M_PI / 2 ) );
+ int dy = KiROUND( m_Height * sin( angle + M_PI / 2 ) );
+ m_crossBarO.x = m_featureLineGO.x + dx;
+ m_crossBarO.y = m_featureLineGO.y + dy;
+ m_crossBarF.x = m_featureLineDO.x + dx;
+ m_crossBarF.y = m_featureLineDO.y + dy;
+
+ m_arrowG1F.x = m_crossBarO.x + arrow_up_X;
+ m_arrowG1F.y = m_crossBarO.y + arrow_up_Y;
+
+ m_arrowG2F.x = m_crossBarO.x + arrow_dw_X;
+ m_arrowG2F.y = m_crossBarO.y + arrow_dw_Y;
+
+ /* The right arrow is symmetrical to the left.
+ * / = -\ and \ = -/
+ */
+ m_arrowD1F.x = m_crossBarF.x - arrow_dw_X;
+ m_arrowD1F.y = m_crossBarF.y - arrow_dw_Y;
+
+ m_arrowD2F.x = m_crossBarF.x - arrow_up_X;
+ m_arrowD2F.y = m_crossBarF.y - arrow_up_Y;
+
+ m_featureLineGF.x = m_crossBarO.x + hx;
+ m_featureLineGF.y = m_crossBarO.y + hy;
+
+ m_featureLineDF.x = m_crossBarF.x + hx;
+ m_featureLineDF.y = m_crossBarF.y + hy;
+
+ // Calculate the better text position and orientation:
+ wxPoint textPos;
+ textPos.x = (m_crossBarF.x + m_featureLineGF.x) / 2;
+ textPos.y = (m_crossBarF.y + m_featureLineGF.y) / 2;
+ m_Text.SetTextPosition( textPos );
+
+ double newAngle = -RAD2DECIDEG( angle );
+
+ NORMALIZE_ANGLE_POS( newAngle );
+
+ if( newAngle > 900 && newAngle < 2700 )
+ newAngle -= 1800;
+
+ m_Text.SetOrientation( newAngle );
+
+ if( !aDoNotChangeText )
+ {
+ m_Value = measure;
+ msg = ::CoordinateToString( m_Value );
+ SetText( msg );
+ }
+}
+
+
+void DIMENSION::Draw( EDA_DRAW_PANEL* panel, wxDC* DC, GR_DRAWMODE mode_color,
+ const wxPoint& offset )
+{
+ EDA_COLOR_T gcolor;
+ BOARD* brd = GetBoard();
+
+ if( brd->IsLayerVisible( m_Layer ) == false )
+ return;
+
+ m_Text.Draw( panel, DC, mode_color, offset );
+
+ gcolor = brd->GetLayerColor( m_Layer );
+
+ GRSetDrawMode( DC, mode_color );
+ DISPLAY_OPTIONS* displ_opts = (DISPLAY_OPTIONS*)panel->GetDisplayOptions();
+ bool filled = displ_opts ? displ_opts->m_DisplayDrawItemsFill : FILLED;
+ int width = m_Width;
+
+ if( filled )
+ {
+ GRLine( panel->GetClipBox(), DC, m_crossBarO + offset,
+ m_crossBarF + offset, width, gcolor );
+ GRLine( panel->GetClipBox(), DC, m_featureLineGO + offset,
+ m_featureLineGF + offset, width, gcolor );
+ GRLine( panel->GetClipBox(), DC, m_featureLineDO + offset,
+ m_featureLineDF + offset, width, gcolor );
+ GRLine( panel->GetClipBox(), DC, m_crossBarF + offset,
+ m_arrowD1F + offset, width, gcolor );
+ GRLine( panel->GetClipBox(), DC, m_crossBarF + offset,
+ m_arrowD2F + offset, width, gcolor );
+ GRLine( panel->GetClipBox(), DC, m_crossBarO + offset,
+ m_arrowG1F + offset, width, gcolor );
+ GRLine( panel->GetClipBox(), DC, m_crossBarO + offset,
+ m_arrowG2F + offset, width, gcolor );
+ }
+ else
+ {
+ GRCSegm( panel->GetClipBox(), DC, m_crossBarO + offset,
+ m_crossBarF + offset, width, gcolor );
+ GRCSegm( panel->GetClipBox(), DC, m_featureLineGO + offset,
+ m_featureLineGF + offset, width, gcolor );
+ GRCSegm( panel->GetClipBox(), DC, m_featureLineDO + offset,
+ m_featureLineDF + offset, width, gcolor );
+ GRCSegm( panel->GetClipBox(), DC, m_crossBarF + offset,
+ m_arrowD1F + offset, width, gcolor );
+ GRCSegm( panel->GetClipBox(), DC, m_crossBarF + offset,
+ m_arrowD2F + offset, width, gcolor );
+ GRCSegm( panel->GetClipBox(), DC, m_crossBarO + offset,
+ m_arrowG1F + offset, width, gcolor );
+ GRCSegm( panel->GetClipBox(), DC, m_crossBarO + offset,
+ m_arrowG2F + offset, width, gcolor );
+ }
+}
+
+
+// see class_cotation.h
+void DIMENSION::GetMsgPanelInfo( std::vector< MSG_PANEL_ITEM >& aList )
+{
+ // for now, display only the text within the DIMENSION using class TEXTE_PCB.
+ m_Text.GetMsgPanelInfo( aList );
+}
+
+
+bool DIMENSION::HitTest( const wxPoint& aPosition ) const
+{
+ if( m_Text.TextHitTest( aPosition ) )
+ return true;
+
+ int dist_max = m_Width / 2;
+
+ // Locate SEGMENTS
+
+ if( TestSegmentHit( aPosition, m_crossBarO, m_crossBarF, dist_max ) )
+ return true;
+
+ if( TestSegmentHit( aPosition, m_featureLineGO, m_featureLineGF, dist_max ) )
+ return true;
+
+ if( TestSegmentHit( aPosition, m_featureLineDO, m_featureLineDF, dist_max ) )
+ return true;
+
+ if( TestSegmentHit( aPosition, m_crossBarF, m_arrowD1F, dist_max ) )
+ return true;
+
+ if( TestSegmentHit( aPosition, m_crossBarF, m_arrowD2F, dist_max ) )
+ return true;
+
+ if( TestSegmentHit( aPosition, m_crossBarO, m_arrowG1F, dist_max ) )
+ return true;
+
+ if( TestSegmentHit( aPosition, m_crossBarO, m_arrowG2F, dist_max ) )
+ return true;
+
+ return false;
+}
+
+
+bool DIMENSION::HitTest( const EDA_RECT& aRect, bool aContained, int aAccuracy ) const
+{
+ EDA_RECT arect = aRect;
+ arect.Inflate( aAccuracy );
+
+ EDA_RECT rect = GetBoundingBox();
+ if( aAccuracy )
+ rect.Inflate( aAccuracy );
+
+ if( aContained )
+ return arect.Contains( rect );
+
+ return arect.Intersects( rect );
+}
+
+
+const EDA_RECT DIMENSION::GetBoundingBox() const
+{
+ EDA_RECT bBox;
+ int xmin, xmax, ymin, ymax;
+
+ bBox = m_Text.GetTextBox( -1 );
+ xmin = bBox.GetX();
+ xmax = bBox.GetRight();
+ ymin = bBox.GetY();
+ ymax = bBox.GetBottom();
+
+ xmin = std::min( xmin, m_crossBarO.x );
+ xmin = std::min( xmin, m_crossBarF.x );
+ ymin = std::min( ymin, m_crossBarO.y );
+ ymin = std::min( ymin, m_crossBarF.y );
+ xmax = std::max( xmax, m_crossBarO.x );
+ xmax = std::max( xmax, m_crossBarF.x );
+ ymax = std::max( ymax, m_crossBarO.y );
+ ymax = std::max( ymax, m_crossBarF.y );
+
+ xmin = std::min( xmin, m_featureLineGO.x );
+ xmin = std::min( xmin, m_featureLineGF.x );
+ ymin = std::min( ymin, m_featureLineGO.y );
+ ymin = std::min( ymin, m_featureLineGF.y );
+ xmax = std::max( xmax, m_featureLineGO.x );
+ xmax = std::max( xmax, m_featureLineGF.x );
+ ymax = std::max( ymax, m_featureLineGO.y );
+ ymax = std::max( ymax, m_featureLineGF.y );
+
+ xmin = std::min( xmin, m_featureLineDO.x );
+ xmin = std::min( xmin, m_featureLineDF.x );
+ ymin = std::min( ymin, m_featureLineDO.y );
+ ymin = std::min( ymin, m_featureLineDF.y );
+ xmax = std::max( xmax, m_featureLineDO.x );
+ xmax = std::max( xmax, m_featureLineDF.x );
+ ymax = std::max( ymax, m_featureLineDO.y );
+ ymax = std::max( ymax, m_featureLineDF.y );
+
+ bBox.SetX( xmin );
+ bBox.SetY( ymin );
+ bBox.SetWidth( xmax - xmin + 1 );
+ bBox.SetHeight( ymax - ymin + 1 );
+
+ bBox.Normalize();
+
+ return bBox;
+}
+
+
+wxString DIMENSION::GetSelectMenuText() const
+{
+ wxString text;
+ text.Printf( _( "Dimension \"%s\" on %s" ),
+ GetChars( GetText() ), GetChars( GetLayerName() ) );
+
+ return text;
+}
+
+
+const BOX2I DIMENSION::ViewBBox() const
+{
+ BOX2I dimBBox = BOX2I( VECTOR2I( GetBoundingBox().GetPosition() ),
+ VECTOR2I( GetBoundingBox().GetSize() ) );
+ dimBBox.Merge( m_Text.ViewBBox() );
+
+ return dimBBox;
+}
+
+
+EDA_ITEM* DIMENSION::Clone() const
+{
+ return new DIMENSION( *this );
+}