|
|
/*
* 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 move_or_drag_track.cpp * @brief Track editing routines to move and drag track segments or node. */
#include <fctsys.h>
#include <class_drawpanel.h>
#include <confirm.h>
#include <pcb_edit_frame.h>
#include <trigo.h>
#include <macros.h>
#include <gr_basic.h>
#include <class_board.h>
#include <pcbnew.h>
#include <drc.h>
#include <drag.h>
#include <pcbnew_id.h>
static void Show_MoveNode( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition, bool aErase ); static void Show_Drag_Track_Segment_With_Cte_Slope( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition, bool aErase ); static void Abort_MoveTrack( EDA_DRAW_PANEL* Panel, wxDC* DC ); static bool InitialiseDragParameters();
static wxPoint PosInit, s_LastPos; static double s_StartSegmentSlope, s_EndSegmentSlope, s_MovingSegmentSlope, s_StartSegment_Yorg, s_EndSegment_Yorg, s_MovingSegment_Yorg; //slope and intercept parameters of lines
bool s_StartPointVertical, s_EndPointVertical, s_MovingSegmentVertical, s_MovingSegmentHorizontal, s_StartPointHorizontal, s_EndPointHorizontal; // vertical or
// horizontal line
// indicators
bool s_StartSegmentPresent, s_EndSegmentPresent;
static PICKED_ITEMS_LIST s_ItemsListPicker;
/** Abort function for drag or move track
*/ static void Abort_MoveTrack( EDA_DRAW_PANEL* aPanel, wxDC* aDC ) { PCB_EDIT_FRAME* frame = (PCB_EDIT_FRAME*) aPanel->GetParent(); BOARD * pcb = frame->GetBoard();
pcb->HighLightOFF(); pcb->PopHighLight();
frame->SetCurItem( NULL ); aPanel->SetMouseCapture( NULL, NULL );
// Undo move and redraw trace segments.
for( unsigned jj=0 ; jj < g_DragSegmentList.size(); jj++ ) { TRACK* track = g_DragSegmentList[jj].m_Track; g_DragSegmentList[jj].RestoreInitialValues(); track->SetState( IN_EDIT, false ); track->ClearFlags(); }
// Clear the undo picker list:
s_ItemsListPicker.ClearListAndDeleteItems(); EraseDragList(); aPanel->Refresh(); }
// Redraw the moved node according to the mouse cursor position
static void Show_MoveNode( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition, bool aErase ) { auto displ_opts = (PCB_DISPLAY_OPTIONS*) aPanel->GetDisplayOptions(); wxPoint moveVector; int tmp = displ_opts->m_DisplayPcbTrackFill; GR_DRAWMODE draw_mode = GR_XOR | GR_HIGHLIGHT;
displ_opts->m_DisplayPcbTrackFill = false;
#ifndef USE_WX_OVERLAY
aErase = true; #else
aErase = false; #endif
// set the new track coordinates
wxPoint Pos = aPanel->GetParent()->GetCrossHairPosition();
moveVector = Pos - s_LastPos; s_LastPos = Pos;
TRACK *track = NULL;
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ ) { track = g_DragSegmentList[ii].m_Track;
if( aErase ) track->Draw( aPanel, aDC, draw_mode );
if( track->GetFlags() & STARTPOINT ) track->SetStart( track->GetStart() + moveVector );
if( track->GetFlags() & ENDPOINT ) track->SetEnd( track->GetEnd() + moveVector );
if( track->Type() == PCB_VIA_T ) track->SetEnd( track->GetStart() );
track->Draw( aPanel, aDC, draw_mode ); }
displ_opts->m_DisplayPcbTrackFill = tmp;
// Display track length
if( track ) { PCB_BASE_FRAME* frame = (PCB_BASE_FRAME*) aPanel->GetParent(); frame->SetMsgPanel( track ); } }
/* drawing the track segment movement
* > s_MovingSegmentSlope slope = moving track segment slope * > s_StartSegmentSlope slope = slope of the segment connected to the start * point of the moving segment * > s_EndSegmentSlope slope = slope of the segment connected to the end point * of the moving segment * * moved segment function : * yt=s_MovingSegmentSlope * x + s_MovingSegment_Yorg * * segment connected to moved segment's start: * y1 = s_StartSegmentSlope * x + s_StartSegment_Yorg * * segment connected to moved segment's end: * y2=s_EndSegmentSlope * x + s_EndSegment_Yorg * * first intersection point will be located at * y1=yt -> * * xi1=(s_MovingSegment_Yorg-s_StartSegment_Yorg)/(s_StartSegmentSlope-s_MovingSegmentSlope) * yi1=s_MovingSegmentSlope*xi1+s_MovingSegment_Yorg * or yi1=s_StartSegmentSlope*xi1+s_MovingSegment_Yorg * * second intersection point * y2=yt -> * * xi2=(s_MovingSegment_Yorg-s_StartSegment_Yorg)/(s_MovingSegmentSlope-s_MovingSegmentSlope) * yi2=s_MovingSegmentSlope*xi2+s_MovingSegment_Yorg * or yi1=s_EndSegmentSlope*xi2+s_MovingSegment_Yorg * !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! * !!!!! special attention to vertical segments because * !!!!! their slope=infinite * !!!!! intersection point will be calculated using the * !!!!! segment intersecting it * !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! * * Slope parameters are computed once, because they can become undetermined * when moving segments * (i.e. when a segment length is 0) and we want keep them constant */ static void Show_Drag_Track_Segment_With_Cte_Slope( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition, bool aErase ) { double xi1 = 0, yi1 = 0, xi2 = 0, yi2 = 0; // calculated intersection points
double tx1, tx2, ty1, ty2; // temporary storage of points
int dx, dy; bool update = true; TRACK* Track; TRACK* tSegmentToStart = NULL, * tSegmentToEnd = NULL;
if( g_DragSegmentList.size() == 0 ) return;
/* get the segments :
* from last to first in list are: * the segment to move * the segment connected to its end point (if exists) * the segment connected to its start point (if exists) */ int ii = g_DragSegmentList.size() - 1; Track = g_DragSegmentList[ii].m_Track;
if( Track == NULL ) return;
ii--;
if( ii >= 0) { if( s_EndSegmentPresent ) { // Get the segment connected to the end point
tSegmentToEnd = g_DragSegmentList[ii].m_Track; ii--; }
if( s_StartSegmentPresent ) { // Get the segment connected to the start point
if( ii >= 0 ) tSegmentToStart = g_DragSegmentList[ii].m_Track; } }
GR_DRAWMODE draw_mode = GR_XOR | GR_HIGHLIGHT;
// Undraw the current moved track segments before modification
#ifndef USE_WX_OVERLAY
// if( erase )
{ Track->Draw( aPanel, aDC, draw_mode );
if( tSegmentToStart ) tSegmentToStart->Draw( aPanel, aDC, draw_mode );
if( tSegmentToEnd ) tSegmentToEnd->Draw( aPanel, aDC, draw_mode ); } #endif
// Compute the new track segment position
wxPoint Pos = aPanel->GetParent()->GetCrossHairPosition();
dx = Pos.x - s_LastPos.x; dy = Pos.y - s_LastPos.y;
// move the line by dx and dy
tx1 = (double) ( Track->GetStart().x + dx ); ty1 = (double) ( Track->GetStart().y + dy ); tx2 = (double) ( Track->GetEnd().x + dx ); ty2 = (double) ( Track->GetEnd().y + dy );
// recalculate the segments new parameters and intersection points
// only the intercept will change, segment slopes does not change
// because we are moving parallel with is initial state
if( !s_MovingSegmentVertical ) s_MovingSegment_Yorg = ty1 - ( s_MovingSegmentSlope * tx1 );
if( ( !s_EndPointVertical ) && ( !s_MovingSegmentVertical ) ) { xi2 = ( s_MovingSegment_Yorg - s_EndSegment_Yorg ) / ( s_EndSegmentSlope - s_MovingSegmentSlope ); } else { if( !s_EndPointVertical ) xi2 = tx2; else { //P1=P2
if( !s_EndPointHorizontal ) xi2 = tx2 - dx; else update = false; } }
if( !s_MovingSegmentVertical ) yi2 = ( s_MovingSegmentSlope * xi2 ) + s_MovingSegment_Yorg; else { if( !s_EndPointVertical ) yi2 = ( s_EndSegmentSlope * xi2 ) + s_EndSegment_Yorg; else { if( !s_EndPointHorizontal ) update = false; else yi2 = ( s_MovingSegmentSlope * xi2 ) + s_MovingSegment_Yorg; } }
if( ( !s_StartPointVertical ) && ( !s_MovingSegmentVertical ) ) { xi1 = ( s_MovingSegment_Yorg - s_StartSegment_Yorg ) / ( s_StartSegmentSlope - s_MovingSegmentSlope ); } else { if( !s_StartPointVertical ) xi1 = tx1; else { //P1=P2
if( !s_StartPointHorizontal ) xi1 = tx1 - dx; else { if( !s_StartPointHorizontal ) update = false; } } }
if( !s_MovingSegmentVertical ) yi1 = ( s_MovingSegmentSlope * xi1 ) + s_MovingSegment_Yorg; else { if( !s_StartPointVertical ) yi1 = ( s_StartSegmentSlope * xi1 ) + s_StartSegment_Yorg; else { if( !s_StartPointHorizontal ) update = false; else yi2 = ( s_MovingSegmentSlope * xi1 ) + s_MovingSegment_Yorg; } }
// update the segment coordinates (if possible)
if( tSegmentToStart == NULL ) { xi1 = tx1; yi1 = ty1; }
if( tSegmentToEnd == NULL ) { xi2 = tx2; yi2 = ty2; }
if( update ) { s_LastPos = Pos; Track->SetStart( wxPoint( KiROUND( xi1 ), KiROUND( yi1 ) ) ); Track->SetEnd( wxPoint( KiROUND( xi2 ), KiROUND( yi2 ) ) );
if( tSegmentToEnd ) { if( tSegmentToEnd->GetFlags() & STARTPOINT ) tSegmentToEnd->SetStart( Track->GetEnd() ); else tSegmentToEnd->SetEnd( Track->GetEnd() ); }
if( tSegmentToStart ) { if( tSegmentToStart->GetFlags() & STARTPOINT ) tSegmentToStart->SetStart( Track->GetStart() ); else tSegmentToStart->SetEnd( Track->GetStart() ); } }
Track->Draw( aPanel, aDC, draw_mode );
if( tSegmentToStart ) tSegmentToStart->Draw( aPanel, aDC, draw_mode );
if( tSegmentToEnd ) tSegmentToEnd->Draw( aPanel, aDC, draw_mode );
// Display track length
PCB_BASE_FRAME* frame = (PCB_BASE_FRAME*) aPanel->GetParent(); frame->SetMsgPanel( Track ); }
/* Init variables (slope, Y intersect point, flags) for
* Show_Drag_Track_Segment_With_Cte_Slope() * return true if Ok, false if dragging is not possible * (2 colinear segments) */ bool InitialiseDragParameters() { double tx1, tx2, ty1, ty2; // temporary storage of points
TRACK* Track; TRACK* tSegmentToStart = NULL, * tSegmentToEnd = NULL;
if( g_DragSegmentList.size() == 0 ) return false;
/* get the segments :
* from last to first in list are: * the segment to move * the segment connected to its end point (if exists) * the segment connected to its start point (if exists) */ int ii = g_DragSegmentList.size() - 1; Track = g_DragSegmentList[ii].m_Track; if( Track == NULL ) return false;
ii--;
if( ii >= 0) { if( s_EndSegmentPresent ) { tSegmentToEnd = g_DragSegmentList[ii].m_Track; // Get the segment connected to
// the end point
ii--; }
if( s_StartSegmentPresent ) { if( ii >= 0 ) tSegmentToStart = g_DragSegmentList[ii].m_Track; // Get the segment connected to
// the start point
} }
// would be nice to eliminate collinear segments here, so we don't
// have to deal with that annoying "Unable to drag this segment: two
// collinear segments"
s_StartPointVertical = false; s_EndPointVertical = false; s_MovingSegmentVertical = false; s_StartPointHorizontal = false; s_EndPointHorizontal = false; s_MovingSegmentHorizontal = false;
// Init parameters for the starting point of the moved segment
if( tSegmentToStart ) { if( tSegmentToStart->GetFlags() & ENDPOINT ) { tx1 = (double) tSegmentToStart->GetStart().x; ty1 = (double) tSegmentToStart->GetStart().y; tx2 = (double) tSegmentToStart->GetEnd().x; ty2 = (double) tSegmentToStart->GetEnd().y; } else { tx1 = (double) tSegmentToStart->GetEnd().x; ty1 = (double) tSegmentToStart->GetEnd().y; tx2 = (double) tSegmentToStart->GetStart().x; ty2 = (double) tSegmentToStart->GetStart().y; } } else // move the start point on a line starting at Track->GetStart(), and perpendicular to Track
{ tx1 = (double) Track->GetStart().x; ty1 = (double) Track->GetStart().y; tx2 = (double) Track->GetEnd().x; ty2 = (double) Track->GetEnd().y; RotatePoint( &tx2, &ty2, tx1, ty1, 900 ); }
if( tx1 != tx2 ) { s_StartSegmentSlope = ( ty2 - ty1 ) / ( tx2 - tx1 ); s_StartSegment_Yorg = ty1 - ( ty2 - ty1 ) * tx1 / ( tx2 - tx1 ); } else { s_StartPointVertical = true; //signal first segment vertical
}
if( ty1 == ty2 ) { s_StartPointHorizontal = true; }
// Init parameters for the ending point of the moved segment
if( tSegmentToEnd ) { //check if second line is vertical
if( tSegmentToEnd->GetFlags() & STARTPOINT ) { tx1 = (double) tSegmentToEnd->GetStart().x; ty1 = (double) tSegmentToEnd->GetStart().y; tx2 = (double) tSegmentToEnd->GetEnd().x; ty2 = (double) tSegmentToEnd->GetEnd().y; } else { tx1 = (double) tSegmentToEnd->GetEnd().x; ty1 = (double) tSegmentToEnd->GetEnd().y; tx2 = (double) tSegmentToEnd->GetStart().x; ty2 = (double) tSegmentToEnd->GetStart().y; } } else // move the start point on a line starting at Track->GetEnd(), and perpendicular to Track
{ tx1 = (double) Track->GetEnd().x; ty1 = (double) Track->GetEnd().y; tx2 = (double) Track->GetStart().x; ty2 = (double) Track->GetStart().y; RotatePoint( &tx2, &ty2, tx1, ty1, -900 ); }
if( tx2 != tx1 ) { s_EndSegmentSlope = ( ty2 - ty1 ) / ( tx2 - tx1 ); s_EndSegment_Yorg = ty1 - ( ty2 - ty1 ) * tx1 / ( tx2 - tx1 ); } else { s_EndPointVertical = true; //signal second segment vertical
}
if( ty1 == ty2 ) { s_EndPointHorizontal = true; }
// Init parameters for the moved segment
tx1 = (double) Track->GetStart().x; ty1 = (double) Track->GetStart().y; tx2 = (double) Track->GetEnd().x; ty2 = (double) Track->GetEnd().y;
if( tx2 != tx1 ) { s_MovingSegmentSlope = ( ty2 - ty1 ) / ( tx2 - tx1 ); } else { s_MovingSegmentVertical = true; // signal vertical line
}
if( ty1 == ty2 ) { s_MovingSegmentHorizontal = true; }
// Test if drag is possible:
if( s_MovingSegmentVertical ) { if( s_EndPointVertical || s_StartPointVertical ) return false; } else { if( !s_EndPointVertical && ( s_MovingSegmentSlope == s_EndSegmentSlope ) ) return false;
if( !s_StartPointVertical && ( s_MovingSegmentSlope == s_StartSegmentSlope ) ) return false; }
return true; }
void PCB_EDIT_FRAME::StartMoveOneNodeOrSegment( TRACK* aTrack, wxDC* aDC, int aCommand ) { if( !aTrack ) return;
EraseDragList();
// Change highlighted net: the new one will be highlighted
GetBoard()->PushHighLight();
if( GetBoard()->IsHighLightNetON() ) HighLight( aDC );
PosInit = GetCrossHairPosition();
if( aTrack->Type() == PCB_VIA_T ) { aTrack->SetFlags( IS_DRAGGED | STARTPOINT | ENDPOINT ); AddSegmentToDragList( aTrack->GetFlags(), aTrack );
if( aCommand != ID_POPUP_PCB_MOVE_TRACK_SEGMENT ) { Collect_TrackSegmentsToDrag( GetBoard(), aTrack->GetStart(), aTrack->GetLayerSet(), aTrack->GetNetCode(), aTrack->GetWidth() / 2 ); }
PosInit = aTrack->GetStart(); } else { STATUS_FLAGS diag = aTrack->IsPointOnEnds( GetCrossHairPosition(), -1 ); wxPoint pos;
switch( aCommand ) { case ID_POPUP_PCB_MOVE_TRACK_SEGMENT: // Move segment
aTrack->SetFlags( IS_DRAGGED | ENDPOINT | STARTPOINT ); AddSegmentToDragList( aTrack->GetFlags(), aTrack ); break;
case ID_POPUP_PCB_DRAG_TRACK_SEGMENT: // drag a segment
pos = aTrack->GetStart(); Collect_TrackSegmentsToDrag( GetBoard(), pos, aTrack->GetLayerSet(), aTrack->GetNetCode(), aTrack->GetWidth() / 2 ); pos = aTrack->GetEnd(); aTrack->SetFlags( IS_DRAGGED | ENDPOINT | STARTPOINT ); Collect_TrackSegmentsToDrag( GetBoard(), pos, aTrack->GetLayerSet(), aTrack->GetNetCode(), aTrack->GetWidth() / 2 ); break;
case ID_POPUP_PCB_MOVE_TRACK_NODE: // Drag via or move node
pos = (diag & STARTPOINT) ? aTrack->GetStart() : aTrack->GetEnd(); Collect_TrackSegmentsToDrag( GetBoard(), pos, aTrack->GetLayerSet(), aTrack->GetNetCode(), aTrack->GetWidth() / 2 ); PosInit = pos; break; }
aTrack->SetFlags( IS_DRAGGED ); }
// Prepare the Undo command
ITEM_PICKER picker( aTrack, UR_CHANGED ); picker.SetLink( aTrack->Clone() ); s_ItemsListPicker.PushItem( picker );
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ ) { TRACK* draggedtrack = g_DragSegmentList[ii].m_Track; picker.SetItem( draggedtrack ); picker.SetLink( draggedtrack->Clone() ); s_ItemsListPicker.PushItem( picker ); draggedtrack = (TRACK*) picker.GetLink(); draggedtrack->SetStatus( 0 ); draggedtrack->ClearFlags(); }
s_LastPos = PosInit; m_canvas->SetMouseCapture( Show_MoveNode, Abort_MoveTrack );
GetBoard()->SetHighLightNet( aTrack->GetNetCode() ); GetBoard()->HighLightON();
GetBoard()->DrawHighLight( m_canvas, aDC, GetBoard()->GetHighLightNetCode() ); m_canvas->CallMouseCapture( aDC, wxDefaultPosition, true );
UndrawAndMarkSegmentsToDrag( m_canvas, aDC ); }
void PCB_EDIT_FRAME::Start_DragTrackSegmentAndKeepSlope( TRACK* track, wxDC* DC ) { TRACK* TrackToStartPoint = NULL; TRACK* TrackToEndPoint = NULL; bool error = false;
if( !track ) return;
// TODO: Use cleanup functions to merge collinear segments if track
// is connected to a collinear segment.
s_StartSegmentPresent = s_EndSegmentPresent = true;
if( ( track->start == NULL ) || ( track->start->Type() == PCB_TRACE_T ) ) TrackToStartPoint = track->GetTrack( GetBoard()->m_Track, NULL, ENDPOINT_START, true, false );
// Test if more than one segment is connected to this point
if( TrackToStartPoint ) { TrackToStartPoint->SetState( BUSY, true );
if( ( TrackToStartPoint->Type() == PCB_VIA_T ) || track->GetTrack( GetBoard()->m_Track, NULL, ENDPOINT_START, true, false ) ) error = true;
TrackToStartPoint->SetState( BUSY, false ); }
if( ( track->end == NULL ) || ( track->end->Type() == PCB_TRACE_T ) ) TrackToEndPoint = track->GetTrack( GetBoard()->m_Track, NULL, ENDPOINT_END, true, false );
// Test if more than one segment is connected to this point
if( TrackToEndPoint ) { TrackToEndPoint->SetState( BUSY, true );
if( (TrackToEndPoint->Type() == PCB_VIA_T) || track->GetTrack( GetBoard()->m_Track, NULL, ENDPOINT_END, true, false ) ) error = true;
TrackToEndPoint->SetState( BUSY, false ); }
if( error ) { DisplayError( this, _( "Unable to drag this segment: too many segments connected" ) ); return; }
if( !TrackToStartPoint || ( TrackToStartPoint->Type() != PCB_TRACE_T ) ) s_StartSegmentPresent = false;
if( !TrackToEndPoint || ( TrackToEndPoint->Type() != PCB_TRACE_T ) ) s_EndSegmentPresent = false;
// Change high light net: the new one will be highlighted
GetBoard()->PushHighLight();
if( GetBoard()->IsHighLightNetON() ) HighLight( DC );
EraseDragList();
track->SetFlags( IS_DRAGGED );
if( TrackToStartPoint ) { STATUS_FLAGS flag = STARTPOINT;
if( track->GetStart() != TrackToStartPoint->GetStart() ) flag = ENDPOINT;
AddSegmentToDragList( flag, TrackToStartPoint ); track->SetFlags( STARTPOINT ); }
if( TrackToEndPoint ) { STATUS_FLAGS flag = STARTPOINT;
if( track->GetEnd() != TrackToEndPoint->GetStart() ) flag = ENDPOINT;
AddSegmentToDragList( flag, TrackToEndPoint ); track->SetFlags( ENDPOINT ); }
AddSegmentToDragList( track->GetFlags(), track );
UndrawAndMarkSegmentsToDrag( m_canvas, DC );
PosInit = GetCrossHairPosition(); s_LastPos = GetCrossHairPosition(); m_canvas->SetMouseCapture( Show_Drag_Track_Segment_With_Cte_Slope, Abort_MoveTrack );
GetBoard()->SetHighLightNet( track->GetNetCode() ); GetBoard()->HighLightON(); GetBoard()->DrawHighLight( m_canvas, DC, GetBoard()->GetHighLightNetCode() );
// Prepare the Undo command
ITEM_PICKER picker( NULL, UR_CHANGED );
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ ) { TRACK* draggedtrack = g_DragSegmentList[ii].m_Track; picker.SetItem( draggedtrack); picker.SetLink ( draggedtrack->Clone() ); s_ItemsListPicker.PushItem( picker ); draggedtrack = (TRACK*) picker.GetLink(); draggedtrack->SetStatus( 0 ); draggedtrack->ClearFlags(); }
if( !InitialiseDragParameters() ) { DisplayError( this, _( "Unable to drag this segment: two collinear segments" ) ); m_canvas->SetMouseCaptureCallback( NULL ); Abort_MoveTrack( m_canvas, DC ); return; } }
// Place a dragged (or moved) track segment or via
bool PCB_EDIT_FRAME::PlaceDraggedOrMovedTrackSegment( TRACK* Track, wxDC* DC ) { int errdrc;
if( Track == NULL ) return false;
int current_net_code = Track->GetNetCode();
// DRC control:
if( Settings().m_legacyDrcOn ) { errdrc = m_drc->DrcOnCreatingTrack( Track, GetBoard()->m_Track );
if( errdrc == BAD_DRC ) return false;
// Test the dragged segments
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ ) { errdrc = m_drc->DrcOnCreatingTrack( g_DragSegmentList[ii].m_Track, GetBoard()->m_Track );
if( errdrc == BAD_DRC ) return false; } }
// DRC Ok: place track segments
Track->ClearFlags(); Track->SetState( IN_EDIT, false );
// Draw dragged tracks
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ ) { Track = g_DragSegmentList[ii].m_Track; Track->SetState( IN_EDIT, false ); Track->ClearFlags();
/* Test the connections modified by the move
* (only pad connection must be tested, track connection will be * tested by TestNetConnection() ) */ LSET layerMask( Track->GetLayer() );
Track->start = GetBoard()->GetPadFast( Track->GetStart(), layerMask );
if( Track->start ) Track->SetState( BEGIN_ONPAD, true ); else Track->SetState( BEGIN_ONPAD, false );
Track->end = GetBoard()->GetPadFast( Track->GetEnd(), layerMask );
if( Track->end ) Track->SetState( END_ONPAD, true ); else Track->SetState( END_ONPAD, false ); }
EraseDragList();
SaveCopyInUndoList( s_ItemsListPicker, UR_UNSPECIFIED ); s_ItemsListPicker.ClearItemsList(); // s_ItemsListPicker is no more owner of picked items
GetBoard()->PopHighLight();
OnModify(); m_canvas->SetMouseCapture( NULL, NULL );
if( current_net_code > 0 ) TestNetConnection( DC, current_net_code );
m_canvas->Refresh();
return true; }
|