/* * KiRouter - a push-and-(sometimes-)shove PCB router * * Copyright (C) 2013-2014 CERN * Author: Tomasz Wlostowski * * 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 . */ #include "pns_dragger.h" #include "pns_shove.h" #include "pns_router.h" PNS_DRAGGER::PNS_DRAGGER( PNS_ROUTER* aRouter ) : PNS_ALGO_BASE( aRouter ) { m_world = NULL; m_lastNode = NULL; m_mode = SEGMENT; m_draggedVia = NULL; m_shove = NULL; m_draggedSegmentIndex = 0; m_dragStatus = false; m_currentMode = RM_MarkObstacles; m_initialVia = NULL; } PNS_DRAGGER::~PNS_DRAGGER() { if( m_shove ) delete m_shove; } void PNS_DRAGGER::SetWorld( PNS_NODE* aWorld ) { m_world = aWorld; } bool PNS_DRAGGER::startDragSegment( const VECTOR2D& aP, PNS_SEGMENT* aSeg ) { int w2 = aSeg->Width() / 2; m_draggedLine = m_world->AssembleLine( aSeg, &m_draggedSegmentIndex ); m_shove->SetInitialLine( m_draggedLine ); m_lastValidDraggedLine = m_draggedLine; m_lastValidDraggedLine.ClearSegmentLinks(); if( ( aP - aSeg->Seg().A ).EuclideanNorm() <= w2 ) m_mode = CORNER; else if( ( aP - aSeg->Seg().B ).EuclideanNorm() <= w2 ) { m_draggedSegmentIndex++; m_mode = CORNER; } else m_mode = SEGMENT; return true; } bool PNS_DRAGGER::startDragVia( const VECTOR2D& aP, PNS_VIA* aVia ) { m_draggedVia = aVia; m_initialVia = aVia; m_mode = VIA; VECTOR2I p0( aVia->Pos() ); PNS_JOINT* jt = m_world->FindJoint( p0, aVia->Layers().Start(), aVia->Net() ); if( !jt ) return false; for( PNS_ITEM* item : jt->LinkList() ) { if( item->OfKind( PNS_ITEM::SEGMENT ) ) { int segIndex; PNS_SEGMENT* seg = ( PNS_SEGMENT*) item; PNS_LINE l = m_world->AssembleLine( seg, &segIndex ); if( segIndex != 0 ) l.Reverse(); m_origViaConnections.Add( l ); } } return true; } bool PNS_DRAGGER::Start( const VECTOR2I& aP, PNS_ITEM* aStartItem ) { m_shove = new PNS_SHOVE( m_world, Router() ); m_lastNode = NULL; m_draggedItems.Clear(); m_currentMode = Settings().Mode(); TRACE( 2, "StartDragging: item %p [kind %d]", aStartItem % aStartItem->Kind() ); switch( aStartItem->Kind() ) { case PNS_ITEM::SEGMENT: return startDragSegment( aP, static_cast( aStartItem ) ); case PNS_ITEM::VIA: return startDragVia( aP, static_cast( aStartItem ) ); default: return false; } } bool PNS_DRAGGER::dragMarkObstacles( const VECTOR2I& aP ) { if( m_lastNode ) { delete m_lastNode; m_lastNode = NULL; } switch( m_mode ) { case SEGMENT: case CORNER: { int thresh = Settings().SmoothDraggedSegments() ? m_draggedLine.Width() / 4 : 0; PNS_LINE dragged( m_draggedLine ); if( m_mode == SEGMENT ) dragged.DragSegment( aP, m_draggedSegmentIndex, thresh ); else dragged.DragCorner( aP, m_draggedSegmentIndex, thresh ); m_lastNode = m_shove->CurrentNode()->Branch(); m_lastValidDraggedLine = dragged; m_lastValidDraggedLine.ClearSegmentLinks(); m_lastValidDraggedLine.Unmark(); m_lastNode->Add( &m_lastValidDraggedLine ); m_draggedItems.Clear(); m_draggedItems.Add( m_lastValidDraggedLine ); break; } case VIA: // fixme... { m_lastNode = m_shove->CurrentNode()->Branch(); dumbDragVia( m_initialVia, m_lastNode, aP ); break; } } if( Settings().CanViolateDRC() ) m_dragStatus = true; else m_dragStatus = !m_world->CheckColliding( m_draggedItems ); return true; } void PNS_DRAGGER::dumbDragVia( PNS_VIA* aVia, PNS_NODE* aNode, const VECTOR2I& aP ) { m_draggedItems.Clear(); // fixme: this is awful. m_draggedVia = aVia->Clone(); m_draggedVia->SetPos( aP ); m_draggedItems.Add( m_draggedVia ); m_lastNode->Remove( aVia ); m_lastNode->Add( m_draggedVia ); for( PNS_ITEM* item : m_origViaConnections.Items() ) { if( const PNS_LINE* l = dyn_cast( item ) ) { PNS_LINE origLine( *l ); PNS_LINE draggedLine( *l ); draggedLine.DragCorner( aP, origLine.CLine().Find( aVia->Pos() ) ); draggedLine.ClearSegmentLinks(); m_draggedItems.Add( draggedLine ); m_lastNode->Remove( &origLine ); m_lastNode->Add( &draggedLine ); } } } bool PNS_DRAGGER::dragShove( const VECTOR2I& aP ) { bool ok = false; if( m_lastNode ) { delete m_lastNode; m_lastNode = NULL; } switch( m_mode ) { case SEGMENT: case CORNER: { int thresh = Settings().SmoothDraggedSegments() ? m_draggedLine.Width() / 4 : 0; PNS_LINE dragged( m_draggedLine ); if( m_mode == SEGMENT ) dragged.DragSegment( aP, m_draggedSegmentIndex, thresh ); else dragged.DragCorner( aP, m_draggedSegmentIndex, thresh ); PNS_SHOVE::SHOVE_STATUS st = m_shove->ShoveLines( dragged ); if( st == PNS_SHOVE::SH_OK ) ok = true; else if( st == PNS_SHOVE::SH_HEAD_MODIFIED ) { dragged = m_shove->NewHead(); ok = true; } m_lastNode = m_shove->CurrentNode()->Branch(); if( ok ) m_lastValidDraggedLine = dragged; m_lastValidDraggedLine.ClearSegmentLinks(); m_lastValidDraggedLine.Unmark(); m_lastNode->Add( &m_lastValidDraggedLine ); m_draggedItems.Clear(); m_draggedItems.Add( m_lastValidDraggedLine ); break; } case VIA: { PNS_VIA* newVia; PNS_SHOVE::SHOVE_STATUS st = m_shove->ShoveDraggingVia( m_draggedVia, aP, &newVia ); if( st == PNS_SHOVE::SH_OK || st == PNS_SHOVE::SH_HEAD_MODIFIED ) ok = true; m_lastNode = m_shove->CurrentNode()->Branch(); if( ok ) { m_draggedVia = newVia; m_draggedItems.Clear(); } break; } } m_dragStatus = ok; return ok; } bool PNS_DRAGGER::FixRoute() { if( m_dragStatus ) { Router()->CommitRouting( CurrentNode() ); return true; } return false; } bool PNS_DRAGGER::Drag( const VECTOR2I& aP ) { switch( m_currentMode ) { case RM_MarkObstacles: return dragMarkObstacles( aP ); case RM_Shove: case RM_Walkaround: case RM_Smart: return dragShove( aP ); default: return false; } } PNS_NODE* PNS_DRAGGER::CurrentNode() const { return m_lastNode; } const PNS_ITEMSET PNS_DRAGGER::Traces() { return m_draggedItems; } PNS_LOGGER* PNS_DRAGGER::Logger() { if( m_shove ) return m_shove->Logger(); return NULL; }