summaryrefslogtreecommitdiff
path: root/pcbnew/router/pns_dragger.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'pcbnew/router/pns_dragger.cpp')
-rw-r--r--pcbnew/router/pns_dragger.cpp336
1 files changed, 336 insertions, 0 deletions
diff --git a/pcbnew/router/pns_dragger.cpp b/pcbnew/router/pns_dragger.cpp
new file mode 100644
index 0000000..a30e7cd
--- /dev/null
+++ b/pcbnew/router/pns_dragger.cpp
@@ -0,0 +1,336 @@
+/*
+ * KiRouter - a push-and-(sometimes-)shove PCB router
+ *
+ * Copyright (C) 2013-2014 CERN
+ * Author: Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
+ *
+ * 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 <http://www.gnu.org/licenses/>.
+ */
+
+#include <boost/foreach.hpp>
+
+#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;
+
+ BOOST_FOREACH( 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<PNS_SEGMENT*>( aStartItem ) );
+
+ case PNS_ITEM::VIA:
+ return startDragVia( aP, static_cast<PNS_VIA*>( 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 );
+
+ BOOST_FOREACH( PNS_ITEM* item, m_origViaConnections.Items() )
+ {
+ if( const PNS_LINE* l = dyn_cast<const PNS_LINE*>( 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;
+}