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.
		
		
		
		
		
			
		
			
				
					
					
						
							1107 lines
						
					
					
						
							34 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							1107 lines
						
					
					
						
							34 KiB
						
					
					
				| /* | |
|  * 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 editrack.cpp | |
|  */ | |
| 
 | |
| #include <fctsys.h> | |
| #include <class_drawpanel.h> | |
| #include <trigo.h> | |
| #include <pcbcommon.h> | |
| #include <wxPcbStruct.h> | |
| #include <colors_selection.h> | |
|  | |
| #include <pcbnew.h> | |
| #include <drc_stuff.h> | |
| #include <protos.h> | |
|  | |
| #include <class_board.h> | |
| #include <class_track.h> | |
| #include <class_zone.h> | |
|  | |
| 
 | |
| static void Abort_Create_Track( EDA_DRAW_PANEL* panel, wxDC* DC ); | |
| void        ShowNewTrackWhenMovingCursor( EDA_DRAW_PANEL* aPanel, wxDC* aDC, | |
|                                           const wxPoint& aPosition, bool aErase ); | |
| static void ComputeBreakPoint( TRACK* track, int n, wxPoint end ); | |
| static void DeleteNullTrackSegments( BOARD* pcb, DLIST<TRACK>& aTrackList ); | |
| static void EnsureEndTrackOnPad( D_PAD* Pad ); | |
| 
 | |
| // A PICKED_ITEMS_LIST to store tracks  which are modified/added/deleted | |
| // during a track edition: | |
| static PICKED_ITEMS_LIST s_ItemsListPicker; | |
| 
 | |
| 
 | |
| /* Function called to abort a track creation | |
|  */ | |
| static void Abort_Create_Track( EDA_DRAW_PANEL* Panel, wxDC* DC ) | |
| { | |
|     PCB_EDIT_FRAME* frame = (PCB_EDIT_FRAME*) Panel->GetParent(); | |
|     BOARD* pcb = frame->GetBoard(); | |
|     TRACK* track = (TRACK*) frame->GetCurItem(); | |
| 
 | |
|     if( track && ( track->Type()==PCB_VIA_T || track->Type()==PCB_TRACE_T ) ) | |
|     { | |
|         // Erase the current drawing | |
|         ShowNewTrackWhenMovingCursor( Panel, DC, wxDefaultPosition, false ); | |
| 
 | |
|         if( pcb->IsHighLightNetON() ) | |
|             frame->HighLight( DC ); | |
| 
 | |
|         pcb->PopHighLight(); | |
| 
 | |
|         if( pcb->IsHighLightNetON() ) | |
|             pcb->DrawHighLight( Panel, DC, pcb->GetHighLightNetCode() ); | |
| 
 | |
|         frame->ClearMsgPanel(); | |
| 
 | |
|         // Undo pending changes (mainly a lock point creation) and clear the | |
|         // undo picker list: | |
|         frame->PutDataInPreviousState( &s_ItemsListPicker, false, false ); | |
|         s_ItemsListPicker.ClearListAndDeleteItems(); | |
| 
 | |
|         // Delete current (new) track | |
|         g_CurrentTrackList.DeleteAll(); | |
|     } | |
| 
 | |
|     frame->SetCurItem( NULL ); | |
| } | |
| 
 | |
| /* | |
|  * This function starts a new track segment. | |
|  * If a new track segment is in progress, ends this current new segment, | |
|  * and created a new one. | |
|  */ | |
| TRACK* PCB_EDIT_FRAME::Begin_Route( TRACK* aTrack, wxDC* aDC ) | |
| { | |
|     TRACK*      TrackOnStartPoint = NULL; | |
|     LAYER_MSK   layerMask = GetLayerMask( GetScreen()->m_Active_Layer ); | |
|     BOARD_CONNECTED_ITEM* LockPoint; | |
|     wxPoint     pos = GetCrossHairPosition(); | |
| 
 | |
|     if( aTrack == NULL )  // Starting a new track segment | |
|     { | |
|         m_canvas->SetMouseCapture( ShowNewTrackWhenMovingCursor, Abort_Create_Track ); | |
| 
 | |
|         // Prepare the undo command info | |
|         s_ItemsListPicker.ClearListAndDeleteItems();  // Should not be necessary, but... | |
|  | |
|         GetBoard()->PushHighLight(); | |
| 
 | |
|         // erase old highlight | |
|         if( GetBoard()->IsHighLightNetON() ) | |
|             HighLight( aDC ); | |
| 
 | |
|         g_CurrentTrackList.PushBack( new TRACK( GetBoard() ) ); | |
|         g_CurrentTrackSegment->SetFlags( IS_NEW ); | |
| 
 | |
|         GetBoard()->SetHighLightNet( 0 ); | |
| 
 | |
|         // Search for a starting point of the new track, a track or pad | |
|         LockPoint = GetBoard()->GetLockPoint( pos, layerMask ); | |
| 
 | |
|         D_PAD* pad = NULL; | |
|         if( LockPoint ) // An item (pad or track) is found | |
|         { | |
|             if( LockPoint->Type() == PCB_PAD_T ) | |
|             { | |
|                 pad = (D_PAD*) LockPoint; | |
| 
 | |
|                 // A pad is found: put the starting point on pad center | |
|                 pos = pad->GetPosition(); | |
|                 GetBoard()->SetHighLightNet( pad->GetNet() ); | |
|             } | |
|             else // A track segment is found | |
|             { | |
|                 TrackOnStartPoint    = (TRACK*) LockPoint; | |
|                 GetBoard()->SetHighLightNet( TrackOnStartPoint->GetNet() ); | |
|                 GetBoard()->CreateLockPoint( pos, TrackOnStartPoint, &s_ItemsListPicker ); | |
|             } | |
|         } | |
|         else | |
|         { | |
|             // Not a starting point, but a filled zone area can exist. This is also a | |
|             // good starting point. | |
|             ZONE_CONTAINER* zone; | |
|             zone = GetBoard()->HitTestForAnyFilledArea( pos, | |
|                                                         GetScreen()-> m_Active_Layer, | |
|                                                         GetScreen()-> m_Active_Layer, | |
|                                                         -1 ); | |
| 
 | |
|             if( zone ) | |
|                 GetBoard()->SetHighLightNet( zone->GetNet() ); | |
|         } | |
| 
 | |
|         DBG( g_CurrentTrackList.VerifyListIntegrity() ); | |
| 
 | |
|         BuildAirWiresTargetsList( LockPoint, wxPoint( 0, 0 ), true ); | |
| 
 | |
|         DBG( g_CurrentTrackList.VerifyListIntegrity() ); | |
| 
 | |
|         GetBoard()->HighLightON(); | |
|         GetBoard()->DrawHighLight( m_canvas, aDC, GetBoard()->GetHighLightNetCode() ); | |
| 
 | |
|         // Display info about track Net class, and init track and vias sizes: | |
|         g_CurrentTrackSegment->SetNet( GetBoard()->GetHighLightNetCode() ); | |
|         GetBoard()->SetCurrentNetClass( g_CurrentTrackSegment->GetNetClassName() ); | |
| 
 | |
|         g_CurrentTrackSegment->SetLayer( GetScreen()->m_Active_Layer ); | |
|         g_CurrentTrackSegment->SetWidth( GetBoard()->GetCurrentTrackWidth() ); | |
| 
 | |
|         if( GetBoard()->GetDesignSettings().m_UseConnectedTrackWidth ) | |
|         { | |
|             if( TrackOnStartPoint && TrackOnStartPoint->Type() == PCB_TRACE_T ) | |
|                 g_CurrentTrackSegment->SetWidth( TrackOnStartPoint->GetWidth()); | |
|         } | |
| 
 | |
|         g_CurrentTrackSegment->SetStart( pos ); | |
|         g_CurrentTrackSegment->SetEnd( pos ); | |
| 
 | |
|         if( pad ) | |
|         { | |
|             g_CurrentTrackSegment->m_PadsConnected.push_back( pad ); | |
|             // Useful to display track length, if the pad has a die length: | |
|             g_CurrentTrackSegment->SetState( BEGIN_ONPAD, true ); | |
|             g_CurrentTrackSegment->start = pad; | |
|         } | |
| 
 | |
|         if( g_TwoSegmentTrackBuild ) | |
|         { | |
|             // Create 2nd segment | |
|             g_CurrentTrackList.PushBack( (TRACK*)g_CurrentTrackSegment->Clone() ); | |
| 
 | |
|             DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
|             g_CurrentTrackSegment->start = g_FirstTrackSegment; | |
|             g_FirstTrackSegment->end     = g_CurrentTrackSegment; | |
| 
 | |
|             g_FirstTrackSegment->SetState( BEGIN_ONPAD | END_ONPAD, false ); | |
|         } | |
| 
 | |
|         DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
|         SetMsgPanel( g_CurrentTrackSegment ); | |
|         SetCurItem( g_CurrentTrackSegment, false ); | |
|         m_canvas->CallMouseCapture( aDC, wxDefaultPosition, false ); | |
| 
 | |
|         if( g_Drc_On ) | |
|         { | |
|             if( BAD_DRC == m_drc->Drc( g_CurrentTrackSegment, GetBoard()->m_Track ) ) | |
|             { | |
|                 return g_CurrentTrackSegment; | |
|             } | |
|         } | |
|     } | |
|     else   // Track in progress : segment coordinates are updated by ShowNewTrackWhenMovingCursor. | |
|     { | |
|         // Test for a D.R.C. error: | |
|         if( g_Drc_On ) | |
|         { | |
|             if( BAD_DRC == m_drc->Drc( g_CurrentTrackSegment, GetBoard()->m_Track ) ) | |
|                 return NULL; | |
| 
 | |
|             // We must handle 2 segments | |
|             if( g_TwoSegmentTrackBuild && g_CurrentTrackSegment->Back() ) | |
|             { | |
|                 if( BAD_DRC == m_drc->Drc( g_CurrentTrackSegment->Back(), GetBoard()->m_Track ) ) | |
|                     return NULL; | |
|             } | |
|         } | |
| 
 | |
|         /* Current track is Ok: current segment is kept, and a new one is | |
|          * created unless the current segment is null, or 2 last are null | |
|          * if this is a 2 segments track build. | |
|          */ | |
|         bool CanCreateNewSegment = true; | |
| 
 | |
|         if( !g_TwoSegmentTrackBuild && g_CurrentTrackSegment->IsNull() ) | |
|             CanCreateNewSegment = false; | |
| 
 | |
|         if( g_TwoSegmentTrackBuild && g_CurrentTrackSegment->IsNull() | |
|           && g_CurrentTrackSegment->Back() | |
|           && g_CurrentTrackSegment->Back()->IsNull() ) | |
|             CanCreateNewSegment = false; | |
| 
 | |
|         if( CanCreateNewSegment ) | |
|         { | |
|             // Erase old track on screen | |
|             DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
|             ShowNewTrackWhenMovingCursor( m_canvas, aDC, wxDefaultPosition, false ); | |
| 
 | |
|             DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
|             if( g_Raccord_45_Auto ) | |
|                 Add45DegreeSegment( aDC ); | |
| 
 | |
|             TRACK* previousTrack = g_CurrentTrackSegment; | |
| 
 | |
|             TRACK* newTrack = (TRACK*)g_CurrentTrackSegment->Clone(); | |
|             g_CurrentTrackList.PushBack( newTrack ); | |
|             newTrack->SetFlags( IS_NEW ); | |
| 
 | |
|             newTrack->SetState( BEGIN_ONPAD | END_ONPAD, false ); | |
| 
 | |
|             D_PAD* pad = GetBoard()->GetPad( previousTrack, FLG_END ); | |
| 
 | |
|             if( pad ) | |
|             { | |
|                 newTrack->m_PadsConnected.push_back( pad ); | |
|                 previousTrack->m_PadsConnected.push_back( pad ); | |
|             } | |
| 
 | |
|             newTrack->start = previousTrack->end; | |
| 
 | |
|             DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
|             newTrack->SetStart( newTrack->GetEnd() ); | |
| 
 | |
|             newTrack->SetLayer( GetScreen()->m_Active_Layer ); | |
| 
 | |
|             if( !GetBoard()->GetDesignSettings().m_UseConnectedTrackWidth ) | |
|                 newTrack->SetWidth( GetBoard()->GetCurrentTrackWidth() ); | |
| 
 | |
|             DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
|             // Show the new position | |
|             ShowNewTrackWhenMovingCursor( m_canvas, aDC, wxDefaultPosition, false ); | |
|         } | |
|     } | |
| 
 | |
|     SetCurItem( g_CurrentTrackSegment, false ); | |
|     return g_CurrentTrackSegment; | |
| } | |
| 
 | |
| 
 | |
| bool PCB_EDIT_FRAME::Add45DegreeSegment( wxDC* aDC ) | |
| { | |
|     int dx0, dy0, dx1, dy1; | |
| 
 | |
|     if( g_CurrentTrackList.GetCount() < 2 ) | |
|         return false;                         // There must be 2 segments. | |
|  | |
|     TRACK* curTrack  = g_CurrentTrackSegment; | |
|     TRACK* prevTrack = curTrack->Back(); | |
| 
 | |
|     // Test if we have 2 consecutive track segments ( not via ) to connect. | |
|     if( curTrack->Type() != PCB_TRACE_T || prevTrack->Type() != PCB_TRACE_T ) | |
|     { | |
|         return false; | |
|     } | |
| 
 | |
|     int segm_step_45 = KiROUND( GetScreen()->GetGridSize().x / 2 ); | |
| 
 | |
|     if( segm_step_45 < ( curTrack->GetWidth() * 2 ) ) | |
|         segm_step_45 = curTrack->GetWidth() * 2; | |
| 
 | |
|     // Test if the segments are horizontal or vertical. | |
|     dx0 = prevTrack->GetEnd().x - prevTrack->GetStart().x; | |
|     dy0 = prevTrack->GetEnd().y - prevTrack->GetStart().y; | |
| 
 | |
|     dx1 = curTrack->GetEnd().x - curTrack->GetStart().x; | |
|     dy1 = curTrack->GetEnd().y - curTrack->GetStart().y; | |
| 
 | |
|     // Segments should have a min length. | |
|     if( std::max( abs( dx0 ), abs( dy0 ) ) < ( segm_step_45 * 2 ) ) | |
|         return false; | |
| 
 | |
|     if( std::max( abs( dx1 ), abs( dy1 ) ) < ( segm_step_45 * 2 ) ) | |
|         return false; | |
| 
 | |
|     // Create a new segment and connect it with the previous 2 segments. | |
|     TRACK* newTrack = (TRACK*)curTrack->Clone(); | |
| 
 | |
|     newTrack->SetStart( prevTrack->GetEnd() ); | |
|     newTrack->SetEnd( curTrack->GetStart() ); | |
| 
 | |
|     if( dx0 == 0 )          // Previous segment is Vertical | |
|     { | |
|         if( dy1 != 0 )      // 2 segments are not 90 degrees. | |
|         { | |
|             delete newTrack; | |
|             return false; | |
|         } | |
| 
 | |
|         /* Calculate coordinates the connection point. | |
|          * The new segment connects the 1st vertical segment and the 2nd | |
|          * horizontal segment. | |
|          */ | |
|         if( dy0 > 0 ) | |
|             newTrack->SetStart( wxPoint(newTrack->GetStart().x, newTrack->GetStart().y -segm_step_45) ); | |
|         else | |
|             newTrack->SetStart( wxPoint(newTrack->GetStart().x, newTrack->GetStart().y + segm_step_45) ); | |
| 
 | |
|         if( dx1 > 0 ) | |
|             newTrack->SetEnd( wxPoint(newTrack->GetEnd().x + segm_step_45, newTrack->GetEnd().y) ); | |
|         else | |
|             newTrack->SetEnd( wxPoint(newTrack->GetEnd().x - segm_step_45, newTrack->GetEnd().y) ); | |
| 
 | |
|         if( g_Drc_On && BAD_DRC == m_drc->Drc( curTrack, GetBoard()->m_Track ) ) | |
|         { | |
|             delete newTrack; | |
|             return false; | |
|         } | |
| 
 | |
|         prevTrack->SetEnd( newTrack->GetStart()); | |
|         curTrack->SetStart( newTrack->GetEnd()); | |
| 
 | |
|         g_CurrentTrackList.Insert( newTrack, curTrack ); | |
|         return true; | |
|     } | |
| 
 | |
|     if( dy0 == 0 )      // Previous segment is horizontal | |
|     { | |
|         if( dx1 != 0 )  // 2 segments are not 90 degrees | |
|         { | |
|             delete newTrack; | |
|             return false; | |
|         } | |
| 
 | |
|         /*  Calculate the coordinates of the point of connection: | |
|          * A new segment has been created, connecting segment 1 | |
|          * (horizontal) and segment 2 (vertical) | |
|          */ | |
|         if( dx0 > 0 ) | |
|             newTrack->SetStart( wxPoint(newTrack->GetStart().x - segm_step_45 , newTrack->GetStart().y)); | |
|         else | |
|             newTrack->SetStart( wxPoint(newTrack->GetStart().x + segm_step_45, newTrack->GetStart().y) ); | |
| 
 | |
|         if( dy1 > 0 ) | |
|             newTrack->SetEnd( wxPoint(newTrack->GetEnd().x, newTrack->GetEnd().y + segm_step_45) ); | |
|         else | |
|             newTrack->SetEnd( wxPoint(newTrack->GetEnd().x, newTrack->GetEnd().y - segm_step_45) ); | |
| 
 | |
|         if( g_Drc_On && BAD_DRC==m_drc->Drc( newTrack, GetBoard()->m_Track ) ) | |
|         { | |
|             delete newTrack; | |
|             return false; | |
|         } | |
| 
 | |
|         prevTrack->SetEnd( newTrack->GetStart()); | |
|         curTrack->SetStart( newTrack->GetEnd()); | |
| 
 | |
|         g_CurrentTrackList.Insert( newTrack, curTrack ); | |
|         return true; | |
|     } | |
| 
 | |
|     return false; | |
| } | |
| 
 | |
| 
 | |
| bool PCB_EDIT_FRAME::End_Route( TRACK* aTrack, wxDC* aDC ) | |
| { | |
|     LAYER_MSK layerMask = GetLayerMask( GetScreen()->m_Active_Layer ); | |
| 
 | |
|     if( aTrack == NULL ) | |
|         return false; | |
| 
 | |
|     if( g_Drc_On && BAD_DRC == m_drc->Drc( g_CurrentTrackSegment, GetBoard()->m_Track ) ) | |
|         return false; | |
| 
 | |
|     // Saving the coordinate of end point of the trace | |
|     wxPoint pos = g_CurrentTrackSegment->GetEnd(); | |
| 
 | |
|     DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
|     if( Begin_Route( aTrack, aDC ) == NULL ) | |
|         return false; | |
| 
 | |
|     ShowNewTrackWhenMovingCursor( m_canvas, aDC, wxDefaultPosition, true ); | |
|     ShowNewTrackWhenMovingCursor( m_canvas, aDC, wxDefaultPosition, false ); | |
|     TraceAirWiresToTargets( aDC ); | |
| 
 | |
|     /* cleanup | |
|      *  if( g_CurrentTrackSegment->Next() != NULL ) | |
|      *  { | |
|      *   delete g_CurrentTrackSegment->Next(); | |
|      *   g_CurrentTrackSegment->SetNext( NULL ); | |
|      *  } | |
|      */ | |
| 
 | |
|     DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
| 
 | |
| 
 | |
|     /* The track here is now chained to the list of track segments. | |
|      * It must be seen in the area of net | |
|      * As close as possible to the segment base (or end), because | |
|      * This helps to reduce the computing time */ | |
| 
 | |
|     // Attaching the end point of the new track to a pad or a track | |
|     BOARD_CONNECTED_ITEM* LockPoint = GetBoard()->GetLockPoint( pos, layerMask ); | |
| 
 | |
|     if( LockPoint ) | |
|     { | |
|         if( LockPoint->Type() ==  PCB_PAD_T )     // End of track is on a pad. | |
|         { | |
|             EnsureEndTrackOnPad( (D_PAD*) LockPoint ); | |
|         } | |
|         else        // If end point of is on a different track, | |
|                     // creates a lock point if not exists | |
|         { | |
|              // Creates a lock point, if not already exists: | |
|             wxPoint hp = g_CurrentTrackSegment->GetEnd(); | |
|             LockPoint = GetBoard()->CreateLockPoint( hp, (TRACK*) LockPoint, &s_ItemsListPicker ); | |
|             g_CurrentTrackSegment->SetEnd(hp); | |
|         } | |
|     } | |
| 
 | |
|     // Delete null length segments: | |
|     DeleteNullTrackSegments( GetBoard(), g_CurrentTrackList ); | |
| 
 | |
|     // Insert new segments if they exist. | |
|     // g_FirstTrackSegment can be NULL on a double click on the starting point | |
|     if( g_FirstTrackSegment != NULL ) | |
|     { | |
|         int    netcode    = g_FirstTrackSegment->GetNet(); | |
|         TRACK* firstTrack = g_FirstTrackSegment; | |
|         int    newCount   = g_CurrentTrackList.GetCount(); | |
| 
 | |
|         // Put entire new current segment list in BOARD, and prepare undo command | |
|         TRACK* track; | |
|         TRACK* insertBeforeMe = g_CurrentTrackSegment->GetBestInsertPoint( GetBoard() ); | |
| 
 | |
|         while( ( track = g_CurrentTrackList.PopFront() ) != NULL ) | |
|         { | |
|             ITEM_PICKER picker( track, UR_NEW ); | |
|             s_ItemsListPicker.PushItem( picker ); | |
|             GetBoard()->m_Track.Insert( track, insertBeforeMe ); | |
|         } | |
| 
 | |
|         TraceAirWiresToTargets( aDC ); | |
| 
 | |
|         int i = 0; | |
| 
 | |
|         for( track = firstTrack; track && i < newCount; ++i, track = track->Next() ) | |
|         { | |
|             track->ClearFlags(); | |
|             track->SetState( BUSY, false ); | |
|         } | |
| 
 | |
|         // delete the old track, if it exists and is redundant | |
|         if( g_AutoDeleteOldTrack ) | |
|         { | |
|             EraseRedundantTrack( aDC, firstTrack, newCount, &s_ItemsListPicker ); | |
|         } | |
| 
 | |
|         SaveCopyInUndoList( s_ItemsListPicker, UR_UNSPECIFIED ); | |
|         s_ItemsListPicker.ClearItemsList(); // s_ItemsListPicker is no more owner of picked items | |
|  | |
|         // compute the new ratsnest | |
|         TestNetConnection( aDC, netcode ); | |
|         OnModify(); | |
|         SetMsgPanel( GetBoard() ); | |
| 
 | |
|         // Redraw the entire new track. | |
|         DrawTraces( m_canvas, aDC, firstTrack, newCount, GR_OR ); | |
|     } | |
| 
 | |
|     wxASSERT( g_FirstTrackSegment == NULL ); | |
|     wxASSERT( g_CurrentTrackSegment == NULL ); | |
|     wxASSERT( g_CurrentTrackList.GetCount() == 0 ); | |
| 
 | |
|     if( GetBoard()->IsHighLightNetON() ) | |
|         HighLight( aDC ); | |
| 
 | |
|     GetBoard()->PopHighLight(); | |
| 
 | |
|     if( GetBoard()->IsHighLightNetON() ) | |
|         GetBoard()->DrawHighLight( m_canvas, aDC, GetBoard()->GetHighLightNetCode() ); | |
| 
 | |
|     m_canvas->SetMouseCapture( NULL, NULL ); | |
|     SetCurItem( NULL ); | |
| 
 | |
|     return true; | |
| } | |
| 
 | |
| 
 | |
| TRACK* LocateIntrusion( TRACK* listStart, TRACK* aTrack, LAYER_NUM aLayer, const wxPoint& aRef ) | |
| { | |
|     int     net   = aTrack->GetNet(); | |
|     int     width = aTrack->GetWidth(); | |
| 
 | |
|     TRACK*  found = NULL; | |
| 
 | |
|     for( TRACK* track = listStart; track; track = track->Next() ) | |
|     { | |
|         if( track->Type() == PCB_TRACE_T )    // skip vias | |
|         { | |
|             if( track->GetState( BUSY | IS_DELETED ) ) | |
|                 continue; | |
| 
 | |
|             if( aLayer != track->GetLayer() ) | |
|                 continue; | |
| 
 | |
|             if( track->GetNet() == net ) | |
|                 continue; | |
| 
 | |
|             // TRACK::HitTest | |
|             int dist = (width + track->GetWidth()) / 2 + aTrack->GetClearance( track ); | |
| 
 | |
|             if( !TestSegmentHit( aRef, track->GetStart(), track->GetEnd(), dist ) ) | |
|                 continue; | |
| 
 | |
|             found = track; | |
| 
 | |
|             // prefer intrusions from the side, not the end | |
|             wxPoint pos = aRef - track->GetStart(); | |
|             wxPoint vec = track->GetEnd() - track->GetStart(); | |
|             double tmp = (double) pos.x * vec.x + (double) pos.y * vec.y; | |
| 
 | |
|             if( tmp >= 0 && tmp <= (double) vec.x * vec.x + (double) vec.y * vec.y ) | |
|                 break; | |
|         } | |
|     } | |
| 
 | |
|     return found; | |
| } | |
| 
 | |
| 
 | |
| /** | |
|  * Function PushTrack | |
|  * detects if the mouse is pointing into a conflicting track. | |
|  * In this case, it tries to push the new track out of the conflicting track's | |
|  * clearance zone. This gives us a cheap mechanism for drawing tracks that | |
|  * tightly follow others, independent of grid settings. | |
|  * | |
|  * KNOWN BUGS: | |
|  * - we do the same sort of search and calculation up to three times: | |
|  *   1) we search for magnetic hits (in controle.cpp) | |
|  *   2) we check if there's a DRC violation in the making (also controle.cpp) | |
|  *   3) we try to fix the DRC violation (here) | |
|  * - if we have a magnetic hit and a DRC violation at the same time, we choose | |
|  *   the magnetic hit instead of solving the violation | |
|  * - should locate conflicting tracks also when we're crossing over them | |
|  */ | |
| static void PushTrack( EDA_DRAW_PANEL* panel ) | |
| { | |
|     PCB_SCREEN* screen = (PCB_SCREEN*) panel->GetParent()->GetScreen(); | |
|     BOARD*      pcb    = ( (PCB_BASE_FRAME*) (panel->GetParent()) )->GetBoard(); | |
|     wxPoint     cursor = panel->GetParent()->GetCrossHairPosition(); | |
|     wxPoint     cv, vec, n; | |
|     TRACK*      track = g_CurrentTrackSegment; | |
|     TRACK*      other; | |
|     double      det; | |
|     int         dist; | |
|     double      f; | |
| 
 | |
|     other = LocateIntrusion( pcb->m_Track, track, screen->m_Active_Layer, panel->GetParent()->RefPos( true ) ); | |
| 
 | |
|     // are we currently pointing into a conflicting trace ? | |
|     if( !other ) | |
|         return; | |
| 
 | |
|     if( other->GetNet() == track->GetNet() ) | |
|         return; | |
| 
 | |
|     cv  = cursor - other->GetStart(); | |
|     vec = other->GetEnd() - other->GetStart(); | |
| 
 | |
|     det = (double) cv.x * vec.y - (double) cv.y * vec.x; | |
| 
 | |
|     // cursor is right at the center of the old track | |
|     if( !det ) | |
|         return; | |
| 
 | |
|     dist = (track->GetWidth() + 1) / 2 + (other->GetWidth() + 1) / 2 + track->GetClearance( other ) + 2; | |
| 
 | |
|     /* | |
|      * DRC wants >, so +1. | |
|      * We may have a quantization error of 1/sqrt(2), so +1 again. | |
|      */ | |
| 
 | |
|     // Vector "n" is perpendicular to "other", pointing towards the cursor. | |
|     if( det > 0 ) | |
|     { | |
|         n.x = vec.y; | |
|         n.y = -vec.x; | |
|     } | |
|     else | |
|     { | |
|         n.x = -vec.y; | |
|         n.y = vec.x; | |
|     } | |
| 
 | |
|     f   = dist / hypot( double(n.x), double(n.y) ); | |
|     n.x = KiROUND( f * n.x ); | |
|     n.y = KiROUND( f * n.y ); | |
| 
 | |
|     wxPoint hp = track->GetEnd(); | |
|     Project( &hp, cursor, other ); | |
|     track->SetEnd( hp + n ); | |
| } | |
| 
 | |
| 
 | |
| //Helper function: Draws Via circle and Via Clearance circle. | |
| inline void DrawViaCirclesWhenEditingNewTrack( EDA_RECT* aPanelClipBox, | |
|                                                wxDC* aDC, const wxPoint& aPos, | |
|                                                int aViaRadius, | |
|                                                int aViaRadiusWithClearence, | |
|                                                EDA_COLOR_T aColor) | |
| { | |
|     //Current viasize clearence circle | |
|     GRCircle( aPanelClipBox, aDC, aPos.x, aPos.y, aViaRadiusWithClearence, aColor ); | |
|     //Current viasize circle | |
|     GRCircle( aPanelClipBox, aDC, aPos.x, aPos.y, aViaRadius, aColor ); | |
| } | |
| 
 | |
| /* Redraw the current track being created when the mouse cursor is moved | |
|  */ | |
| void ShowNewTrackWhenMovingCursor( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition, | |
|                                    bool aErase ) | |
| { | |
| //    DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
|  | |
|     PCB_SCREEN*     screen = (PCB_SCREEN*) aPanel->GetScreen(); | |
|     PCB_BASE_FRAME* frame  = (PCB_BASE_FRAME*) aPanel->GetParent(); | |
| 
 | |
|     bool      Track_fill_copy = DisplayOpt.DisplayPcbTrackFill; | |
|     DisplayOpt.DisplayPcbTrackFill = true; | |
|     TRACE_CLEARANCE_DISPLAY_MODE_T showTrackClearanceMode = DisplayOpt.ShowTrackClearanceMode; | |
| 
 | |
|     if ( g_FirstTrackSegment == NULL ) | |
|         return; | |
| 
 | |
|     NETCLASS* netclass = g_FirstTrackSegment->GetNetClass(); | |
| 
 | |
|     if( showTrackClearanceMode != DO_NOT_SHOW_CLEARANCE ) | |
|         DisplayOpt.ShowTrackClearanceMode = SHOW_CLEARANCE_ALWAYS; | |
| 
 | |
|     // Values to Via circle | |
|     int boardViaRadius = frame->GetBoard()->GetCurrentViaSize()/2; | |
|     int viaRadiusWithClearence = boardViaRadius+netclass->GetClearance(); | |
|     EDA_RECT* panelClipBox=aPanel->GetClipBox(); | |
| 
 | |
| #ifndef USE_WX_OVERLAY | |
|     // Erase old track | |
|     if( aErase ) | |
|     { | |
|         DrawTraces( aPanel, aDC, g_FirstTrackSegment, g_CurrentTrackList.GetCount(), GR_XOR ); | |
| 
 | |
|         frame->TraceAirWiresToTargets( aDC ); | |
| 
 | |
|         if( showTrackClearanceMode >= SHOW_CLEARANCE_NEW_TRACKS_AND_VIA_AREAS ) | |
|         { | |
|             EDA_COLOR_T color = g_ColorsSettings.GetLayerColor( g_CurrentTrackSegment->GetLayer() ); | |
|             DrawViaCirclesWhenEditingNewTrack( panelClipBox, aDC, g_CurrentTrackSegment->GetEnd(), | |
|                                                boardViaRadius, viaRadiusWithClearence, color); | |
|         } | |
|     } | |
| #endif | |
|     // MacOSX seems to need this. | |
|     if( g_CurrentTrackList.GetCount() == 0 ) | |
|         return; | |
| 
 | |
|     // Set track parameters, that can be modified while creating the track | |
|     g_CurrentTrackSegment->SetLayer( screen->m_Active_Layer ); | |
| 
 | |
|     if( !frame->GetBoard()->GetDesignSettings().m_UseConnectedTrackWidth ) | |
|         g_CurrentTrackSegment->SetWidth( frame->GetBoard()->GetCurrentTrackWidth() ); | |
| 
 | |
|     if( g_TwoSegmentTrackBuild ) | |
|     { | |
|         TRACK* previous_track = g_CurrentTrackSegment->Back(); | |
| 
 | |
|         if( previous_track  &&  previous_track->Type()==PCB_TRACE_T ) | |
|         { | |
|             previous_track->SetLayer( screen->m_Active_Layer ); | |
| 
 | |
|             if( !frame->GetBoard()->GetDesignSettings().m_UseConnectedTrackWidth ) | |
|                 previous_track->SetWidth( frame->GetBoard()->GetCurrentTrackWidth() ); | |
|         } | |
|     } | |
| 
 | |
|     if( g_Track_45_Only_Allowed ) | |
|     { | |
|         if( g_TwoSegmentTrackBuild ) | |
|         { | |
|             g_CurrentTrackSegment->SetEnd( frame->GetCrossHairPosition() ); | |
| 
 | |
|             if( g_Drc_On ) | |
|                 PushTrack( aPanel ); | |
| 
 | |
|             ComputeBreakPoint( g_CurrentTrackSegment, | |
|                                g_CurrentTrackList.GetCount(), | |
|                                g_CurrentTrackSegment->GetEnd() ); | |
|         } | |
|         else | |
|         { | |
|             /* Calculate of the end of the path for the permitted directions: | |
|              * horizontal, vertical or 45 degrees. | |
|              */ | |
|             wxPoint hp = g_CurrentTrackSegment->GetEnd(); | |
|             CalculateSegmentEndPoint( frame->GetCrossHairPosition(), | |
|                                       g_CurrentTrackSegment->GetStart().x, | |
|                                       g_CurrentTrackSegment->GetStart().y, | |
|                                       &hp.x, | |
|                                       &hp.y ); | |
|             g_CurrentTrackSegment->SetEnd(hp); | |
|         } | |
|     } | |
|     else    // Here the angle is arbitrary | |
|     { | |
|         g_CurrentTrackSegment->SetEnd( frame->GetCrossHairPosition() ); | |
|     } | |
| 
 | |
|     // Redraw the new track | |
|     DBG( g_CurrentTrackList.VerifyListIntegrity(); ); | |
|     DrawTraces( aPanel, aDC, g_FirstTrackSegment, g_CurrentTrackList.GetCount(), GR_XOR ); | |
| 
 | |
|     if( showTrackClearanceMode >= SHOW_CLEARANCE_NEW_TRACKS_AND_VIA_AREAS ) | |
|     { | |
|         EDA_COLOR_T color = g_ColorsSettings.GetLayerColor(g_CurrentTrackSegment->GetLayer()); | |
| 
 | |
|         //Via diameter must have taken what we are using, rather than netclass value. | |
|         DrawViaCirclesWhenEditingNewTrack( panelClipBox, aDC, g_CurrentTrackSegment->GetEnd(), | |
|                                            boardViaRadius, viaRadiusWithClearence, color); | |
| 
 | |
|     } | |
| 
 | |
|     /* Display info about current segment and the full new track: | |
|      *  Choose the interesting segment: because we are using a 2 segments step, | |
|      *  the last segment can be null, and the previous segment can be the | |
|      *  interesting segment. | |
|      */ | |
|     TRACK* isegm = g_CurrentTrackSegment; | |
| 
 | |
|     if( isegm->GetLength() == 0 && g_CurrentTrackSegment->Back() ) | |
|         isegm = g_CurrentTrackSegment->Back(); | |
| 
 | |
|     // display interesting segment info only: | |
|     frame->SetMsgPanel( isegm ); | |
| 
 | |
|     // Display current track length (on board) and the the actual track len | |
|     // if there is an extra len due to the len die on the starting pad (if any) | |
|     double   trackLen = 0.0; | |
|     double   lenPadToDie = 0.0; | |
|     wxString msg; | |
| 
 | |
|     // If the starting point is on a pad, add current track length+ length die | |
|     if( g_FirstTrackSegment->GetState( BEGIN_ONPAD ) ) | |
|     { | |
|         D_PAD * pad = (D_PAD *) g_FirstTrackSegment->start; | |
|         lenPadToDie = (double) pad->GetPadToDieLength(); | |
|     } | |
| 
 | |
|     // calculate track len on board: | |
|     for( TRACK* track = g_FirstTrackSegment; track; track = track->Next() ) | |
|         trackLen += track->GetLength(); | |
| 
 | |
|     msg = frame->LengthDoubleToString( trackLen ); | |
|     frame->AppendMsgPanel( _( "Track Len" ), msg, DARKCYAN ); | |
| 
 | |
|     if( lenPadToDie != 0 )      // display the track len on board and the actual track len | |
|     { | |
|         frame->AppendMsgPanel( _( "Full Len" ), msg, DARKCYAN ); | |
|         msg = frame->LengthDoubleToString( trackLen+lenPadToDie ); | |
|         frame->AppendMsgPanel( _( "Pad to die" ), msg, DARKCYAN ); | |
|     } | |
| 
 | |
|     // Add current segments count (number of segments in this new track): | |
|     msg.Printf( wxT( "%d" ), g_CurrentTrackList.GetCount() ); | |
|     frame->AppendMsgPanel( _( "Segs Count" ), msg, DARKCYAN ); | |
| 
 | |
|     DisplayOpt.ShowTrackClearanceMode = showTrackClearanceMode; | |
|     DisplayOpt.DisplayPcbTrackFill    = Track_fill_copy; | |
| 
 | |
|     frame->BuildAirWiresTargetsList( NULL, g_CurrentTrackSegment->GetEnd(), false ); | |
|     frame->TraceAirWiresToTargets( aDC ); | |
| } | |
| 
 | |
| 
 | |
| /* Determine the coordinate to advanced the the current segment | |
|  * in 0, 90, or 45 degrees, depending on position of origin and \a aPosition. | |
|  */ | |
| void CalculateSegmentEndPoint( const wxPoint& aPosition, int ox, int oy, int* fx, int* fy ) | |
| { | |
|     int deltax, deltay, angle; | |
| 
 | |
|     deltax = aPosition.x - ox; | |
|     deltay = aPosition.y - oy; | |
| 
 | |
|     deltax = abs( deltax ); | |
|     deltay = abs( deltay ); | |
|     angle  = 45; | |
| 
 | |
|     if( deltax >= deltay ) | |
|     { | |
|         if( deltax == 0 ) | |
|             angle = 0; | |
|         else if( ( (deltay << 6 ) / deltax ) < 26 ) | |
|             angle = 0; | |
|     } | |
|     else | |
|     { | |
|         angle = 45; | |
| 
 | |
|         if( deltay == 0 ) | |
|             angle = 90; | |
|         else if( ( (deltax << 6 ) / deltay ) < 26 ) | |
|             angle = 90; | |
|     } | |
| 
 | |
|     switch( angle ) | |
|     { | |
|     case 0: | |
|         *fx = aPosition.x; | |
|         *fy = oy; | |
|         break; | |
| 
 | |
|     case 45: | |
|         deltax = std::min( deltax, deltay ); | |
|         deltay = deltax; | |
| 
 | |
|         // Recalculate the signs for deltax and deltaY. | |
|         if( ( aPosition.x - ox ) < 0 ) | |
|             deltax = -deltax; | |
| 
 | |
|         if( ( aPosition.y - oy ) < 0 ) | |
|             deltay = -deltay; | |
| 
 | |
|         *fx = ox + deltax; | |
|         *fy = oy + deltay; | |
|         break; | |
| 
 | |
|     case 90: | |
|         *fx = ox; | |
|         *fy = aPosition.y; | |
|         break; | |
|     } | |
| } | |
| 
 | |
| 
 | |
| /** | |
|  * Compute new track angle based on previous track. | |
|  */ | |
| void ComputeBreakPoint( TRACK* track, int SegmentCount, wxPoint end ) | |
| { | |
|     int iDx    = 0; | |
|     int iDy    = 0; | |
|     int iAngle = 0; | |
| 
 | |
|     if( SegmentCount <= 0 ) | |
|         return; | |
| 
 | |
|     if( track == NULL ) | |
|         return; | |
| 
 | |
|     TRACK* newTrack = track; | |
|     track = track->Back(); | |
|     SegmentCount--; | |
| 
 | |
|     if( track ) | |
|     { | |
|         iDx = end.x - track->GetStart().x; | |
|         iDy = end.y - track->GetStart().y; | |
| 
 | |
|         iDx = abs( iDx ); | |
|         iDy = abs( iDy ); | |
|     } | |
| 
 | |
|     TRACK* lastTrack = track ? track->Back() : NULL; | |
| 
 | |
|     if( lastTrack ) | |
|     { | |
|         if(( (lastTrack->GetEnd().x == lastTrack->GetStart().x) | |
|            || (lastTrack->GetEnd().y == lastTrack->GetStart().y) ) | |
|         && !g_Alternate_Track_Posture) | |
|         { | |
|             iAngle = 45; | |
|         } | |
|     } | |
|     else | |
|     { | |
|         if( g_Alternate_Track_Posture ) | |
|         { | |
|             iAngle = 45; | |
|         } | |
|     } | |
| 
 | |
|     if( iAngle == 0 ) | |
|     { | |
|         if( iDx >= iDy ) | |
|             iAngle = 0; | |
|         else | |
|             iAngle = 90; | |
|     } | |
| 
 | |
|     if( track == NULL ) | |
|         iAngle = -1; | |
| 
 | |
|     switch( iAngle ) | |
|     { | |
|     case -1: | |
|         break; | |
| 
 | |
|     case 0: | |
|         if( ( end.x - track->GetStart().x ) < 0 ) | |
|             track->SetEnd(wxPoint( end.x + iDy, track->GetStart().y)); | |
|         else | |
|             track->SetEnd(wxPoint( end.x - iDy, track->GetStart().y)); | |
|         break; | |
| 
 | |
|     case 45: | |
|         iDx = std::min( iDx, iDy ); | |
|         iDy = iDx; | |
| 
 | |
|         // Recalculate the signs for deltax and deltaY. | |
|         if( ( end.x - track->GetStart().x ) < 0 ) | |
|             iDx = -iDx; | |
| 
 | |
|         if( ( end.y - track->GetStart().y ) < 0 ) | |
|             iDy = -iDy; | |
| 
 | |
|         track->SetEnd(wxPoint(track->GetStart().x + iDx, track->GetStart().y + iDy)); | |
|         break; | |
| 
 | |
|     case 90: | |
|         if( ( end.y - track->GetStart().y ) < 0 ) | |
|             track->SetEnd(wxPoint(track->GetStart().x , end.y + iDx)); | |
|         else | |
|             track->SetEnd(wxPoint(track->GetStart().x , end.y - iDx)); | |
|         break; | |
|     } | |
| 
 | |
|     if( track ) | |
|     { | |
|         if( track->IsNull() ) | |
|             track->SetEnd( end ); | |
| 
 | |
|         newTrack->SetStart( track->GetEnd() ); | |
|     } | |
| 
 | |
|     newTrack->SetEnd( end ); | |
| } | |
| 
 | |
| 
 | |
| /* Delete track segments which have len = 0 after creating a new track | |
|  *  return a pointer on the first segment (start of track list) | |
|  */ | |
| void DeleteNullTrackSegments( BOARD* pcb, DLIST<TRACK>& aTrackList ) | |
| { | |
|     if( aTrackList.GetCount() == 0 ) | |
|         return; | |
| 
 | |
|     TRACK*      track = aTrackList.GetFirst(); | |
|     TRACK*      firsttrack = track; | |
|     TRACK*      oldtrack; | |
| 
 | |
|     BOARD_CONNECTED_ITEM* LockPoint = track->start; | |
| 
 | |
|     while( track != NULL ) | |
|     { | |
|         oldtrack = track; | |
|         track    = track->Next(); | |
| 
 | |
|         if( !oldtrack->IsNull() ) | |
|         { | |
|             continue; | |
|         } | |
| 
 | |
|         // NULL segment, delete it | |
|         if( firsttrack == oldtrack ) | |
|             firsttrack = track; | |
| 
 | |
|         delete aTrackList.Remove( oldtrack ); | |
|     } | |
| 
 | |
|     if( aTrackList.GetCount() == 0 ) | |
|         return;         // all the new track segments have been deleted | |
|  | |
|     // we must set the pointers on connected items and the connection status | |
|     oldtrack = track = firsttrack; | |
|     firsttrack->start = NULL; | |
| 
 | |
|     while( track != NULL ) | |
|     { | |
|         oldtrack = track; | |
|         track    = track->Next(); | |
|         oldtrack->end = track; | |
| 
 | |
|         if( track ) | |
|             track->start = oldtrack; | |
| 
 | |
|         oldtrack->SetStatus( 0 ); | |
|     } | |
| 
 | |
|     firsttrack->start = LockPoint; | |
| 
 | |
|     if( LockPoint && LockPoint->Type()==PCB_PAD_T ) | |
|         firsttrack->SetState( BEGIN_ONPAD, true ); | |
| 
 | |
|     track = firsttrack; | |
| 
 | |
|     while( track != NULL ) | |
|     { | |
|         TRACK* next_track = track->Next(); | |
|         LockPoint = pcb->GetPad( track, FLG_END ); | |
| 
 | |
|         if( LockPoint ) | |
|         { | |
|             track->end = LockPoint; | |
|             track->SetState( END_ONPAD, true ); | |
| 
 | |
|             if( next_track ) | |
|             { | |
|                 next_track->start = LockPoint; | |
|                 next_track->SetState( BEGIN_ONPAD, true ); | |
|             } | |
|         } | |
| 
 | |
|         track = next_track; | |
|     } | |
| } | |
| 
 | |
| 
 | |
| /* Ensure the end point of g_CurrentTrackSegment is on the pad "Pad" | |
|  *  if no, create a new track segment if necessary | |
|  *  and move current (or new) end segment on pad | |
|  */ | |
| void EnsureEndTrackOnPad( D_PAD* aPad ) | |
| { | |
|     if( g_CurrentTrackSegment->GetEnd() == aPad->GetPosition() ) // Ok ! | |
|     { | |
|         g_CurrentTrackSegment->end = aPad; | |
|         g_CurrentTrackSegment->SetState( END_ONPAD, true ); | |
|         return; | |
|     } | |
| 
 | |
|     TRACK* lasttrack = g_CurrentTrackSegment; | |
| 
 | |
|     if( !g_CurrentTrackSegment->IsNull() ) | |
|     { | |
|         // Must create a new segment, from track end to pad center | |
|         g_CurrentTrackList.PushBack( (TRACK*)lasttrack->Clone() ); | |
| 
 | |
|         lasttrack->end = g_CurrentTrackSegment; | |
|     } | |
| 
 | |
|     g_CurrentTrackSegment->SetEnd( aPad->GetPosition() ); | |
|     g_CurrentTrackSegment->SetState( END_ONPAD, false ); | |
| 
 | |
|     g_CurrentTrackSegment->end = aPad; | |
|     g_CurrentTrackSegment->SetState( END_ONPAD, true ); | |
| }
 |