eeschema: Clean bits of DLIST

Removes the extra DLIST manipulation from SCH_LINE_WIRE_BUS_TOOL and
moves the static storage into class storage
This commit is contained in:
Seth Hillbrand 2019-12-11 05:22:59 -08:00
parent ac7538cc22
commit cda291adae
5 changed files with 173 additions and 138 deletions

View File

@ -211,7 +211,7 @@ bool SCH_EDIT_FRAME::SchematicCleanUp( SCH_SCREEN* aScreen )
else if( secondLine->IsEndPoint( firstLine->GetEndPoint() ) ) else if( secondLine->IsEndPoint( firstLine->GetEndPoint() ) )
needed = aScreen->IsJunctionNeeded( firstLine->GetEndPoint() ); needed = aScreen->IsJunctionNeeded( firstLine->GetEndPoint() );
if( !needed && ( line = (SCH_LINE*) secondLine->MergeOverlap( firstLine ) ) ) if( !needed && ( line = secondLine->MergeOverlap( firstLine ) ) )
{ {
remove_item( item ); remove_item( item );
remove_item( secondItem ); remove_item( secondItem );
@ -367,7 +367,7 @@ void SCH_EDIT_FRAME::DeleteJunction( SCH_ITEM* aJunction, bool aAppend )
} }
// Try to merge the remaining lines // Try to merge the remaining lines
if( SCH_LINE* line = (SCH_LINE*) secondLine->MergeOverlap( firstLine ) ) if( SCH_LINE* line = secondLine->MergeOverlap( firstLine ) )
{ {
remove_item( item ); remove_item( item );
remove_item( secondItem ); remove_item( secondItem );

View File

@ -405,7 +405,7 @@ bool SCH_LINE::IsParallel( SCH_LINE* aLine )
} }
EDA_ITEM* SCH_LINE::MergeOverlap( SCH_LINE* aLine ) SCH_LINE* SCH_LINE::MergeOverlap( SCH_LINE* aLine )
{ {
auto less = []( const wxPoint& lhs, const wxPoint& rhs ) -> bool auto less = []( const wxPoint& lhs, const wxPoint& rhs ) -> bool
{ {
@ -418,7 +418,7 @@ EDA_ITEM* SCH_LINE::MergeOverlap( SCH_LINE* aLine )
wxT( "Cannot test line segment for overlap." ) ); wxT( "Cannot test line segment for overlap." ) );
if( this == aLine || GetLayer() != aLine->GetLayer() ) if( this == aLine || GetLayer() != aLine->GetLayer() )
return NULL; return nullptr;
auto leftmost_start = aLine->m_start; auto leftmost_start = aLine->m_start;
auto leftmost_end = aLine->m_end; auto leftmost_end = aLine->m_end;
@ -455,7 +455,7 @@ EDA_ITEM* SCH_LINE::MergeOverlap( SCH_LINE* aLine )
// If we end one before the beginning of the other, no overlap is possible // If we end one before the beginning of the other, no overlap is possible
if( less( leftmost_end, other_start ) ) if( less( leftmost_end, other_start ) )
{ {
return NULL; return nullptr;
} }
// Search for a common end: // Search for a common end:
@ -514,7 +514,7 @@ EDA_ITEM* SCH_LINE::MergeOverlap( SCH_LINE* aLine )
return ret; return ret;
} }
return NULL; return nullptr;
} }

View File

@ -163,7 +163,7 @@ public:
* @param aLine - Line to compare. * @param aLine - Line to compare.
* @return New line that combines the two or NULL on non-overlapping segments. * @return New line that combines the two or NULL on non-overlapping segments.
*/ */
EDA_ITEM* MergeOverlap( SCH_LINE* aLine ); SCH_LINE* MergeOverlap( SCH_LINE* aLine );
/** /**
* Check if two lines are in the same quadrant as each other, using a reference point as * Check if two lines are in the same quadrant as each other, using a reference point as

View File

@ -22,27 +22,63 @@
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
*/ */
#include <connection_graph.h>
#include <sch_line_wire_bus_tool.h> #include <sch_line_wire_bus_tool.h>
#include <ee_selection_tool.h>
#include <ee_actions.h> #include <boost/optional/optional.hpp>
#include <sch_edit_frame.h> #include <wx/debug.h>
#include <sch_view.h> #include <wx/gdicmn.h>
#include <class_draw_panel_gal.h> #include <wx/menu.h>
#include <id.h> #include <wx/string.h>
#include <eeschema_id.h> #include <wx/stringimpl.h>
#include <confirm.h> #include <wx/translation.h>
#include <view/view_group.h> #include <algorithm>
#include <view/view_controls.h> #include <cstdlib>
#include <view/view.h> #include <deque>
#include <tool/tool_manager.h> #include <iterator>
#include <sch_junction.h> #include <memory>
#include <sch_line.h> #include <string>
#include <sch_bus_entry.h> #include <utility>
#include <sch_text.h> #include <vector>
#include <sch_sheet.h>
#include <bitmaps.h>
#include <advanced_config.h> #include <advanced_config.h>
#include "ee_point_editor.h" #include <base_screen.h>
#include <base_struct.h>
#include <core/typeinfo.h>
#include <eda_text.h>
#include <layers_id_colors_and_visibility.h>
#include <math/vector2d.h>
#include <tool/actions.h>
#include <tool/conditional_menu.h>
#include <tool/selection.h>
#include <tool/selection_conditions.h>
#include <tool/tool_action.h>
#include <tool/tool_event.h>
#include <tool/tool_interactive.h>
#include <tool/tool_manager.h>
#include <trigo.h>
#include <undo_redo_container.h>
#include <view/view_controls.h>
#include <connection_graph.h>
#include <eeschema_id.h>
#include <general.h>
#include <sch_bus_entry.h>
#include <sch_connection.h>
#include <sch_draw_panel.h>
#include <sch_edit_frame.h>
#include <sch_item.h>
#include <sch_line.h>
#include <sch_screen.h>
#include <sch_sheet.h>
#include <sch_sheet_path.h>
#include <sch_text.h>
#include <sch_view.h>
#include <ee_actions.h>
#include <ee_point_editor.h>
#include <ee_selection.h>
#include <ee_selection_tool.h>
class BUS_UNFOLD_MENU : public ACTION_MENU class BUS_UNFOLD_MENU : public ACTION_MENU
{ {
@ -253,7 +289,6 @@ bool SCH_LINE_WIRE_BUS_TOOL::IsDrawingLineWireOrBus( const SELECTION& aSelection
int SCH_LINE_WIRE_BUS_TOOL::DrawSegments( const TOOL_EVENT& aEvent ) int SCH_LINE_WIRE_BUS_TOOL::DrawSegments( const TOOL_EVENT& aEvent )
{ {
SCH_LAYER_ID layer = aEvent.Parameter<SCH_LAYER_ID>(); SCH_LAYER_ID layer = aEvent.Parameter<SCH_LAYER_ID>();
SCH_LINE* segment = nullptr;
if( aEvent.HasPosition() ) if( aEvent.HasPosition() )
getViewControls()->WarpCursor( getViewControls()->GetCursorPosition(), true ); getViewControls()->WarpCursor( getViewControls()->GetCursorPosition(), true );
@ -264,10 +299,10 @@ int SCH_LINE_WIRE_BUS_TOOL::DrawSegments( const TOOL_EVENT& aEvent )
if( aEvent.HasPosition() ) if( aEvent.HasPosition() )
{ {
VECTOR2D cursorPos = getViewControls()->GetCursorPosition( !aEvent.Modifier( MD_ALT ) ); VECTOR2D cursorPos = getViewControls()->GetCursorPosition( !aEvent.Modifier( MD_ALT ) );
segment = startSegments( layer, cursorPos ); startSegments( layer, cursorPos );
} }
return doDrawSegments( tool, layer, segment ); return doDrawSegments( tool, layer );
} }
@ -314,7 +349,7 @@ int SCH_LINE_WIRE_BUS_TOOL::UnfoldBus( const TOOL_EVENT& aEvent )
// If we have an unfolded wire to draw, then draw it // If we have an unfolded wire to draw, then draw it
if( segment ) if( segment )
return doDrawSegments( tool, LAYER_WIRE, segment ); return doDrawSegments( tool, LAYER_WIRE );
else else
{ {
m_frame->PopTool( tool ); m_frame->PopTool( tool );
@ -349,10 +384,6 @@ SCH_LINE* SCH_LINE_WIRE_BUS_TOOL::doUnfoldBus( const wxString& aNet )
} }
// Storage for the line segments while drawing
static DLIST<SCH_LINE> s_wires;
/** /**
* A helper function to find any sheet pins at the specified position. * A helper function to find any sheet pins at the specified position.
*/ */
@ -381,22 +412,25 @@ static const SCH_SHEET_PIN* getSheetPin( SCH_SCREEN* aScreen, const wxPoint& aPo
* computes the middle coordinate for 2 segments from the start point to \a aPosition * computes the middle coordinate for 2 segments from the start point to \a aPosition
* with the segments kept in the horizontal or vertical axis only. * with the segments kept in the horizontal or vertical axis only.
* *
* @param aSegment A pointer to a #SCH_LINE object containing the first line break point * @param aSegments A pair of pointers to a #SCH_LINE objects containing the first line break point
* to compute. * to compute.
* @param aPosition A reference to a wxPoint object containing the coordinates of the * @param aPosition A reference to a wxPoint object containing the coordinates of the
* position used to calculate the line break point. * position used to calculate the line break point.
*/ */
static void computeBreakPoint( SCH_SCREEN* aScreen, SCH_LINE* aSegment, wxPoint& aPosition ) static void computeBreakPoint( SCH_SCREEN* aScreen, std::pair<SCH_LINE*, SCH_LINE*> aSegments,
wxPoint& aPosition )
{ {
wxCHECK_RET( aSegment != nullptr, wxT( "Cannot compute break point of NULL line segment." ) ); wxCHECK_RET( aSegments.first && aSegments.second,
wxT( "Cannot compute break point of NULL line segment." ) );
SCH_LINE* nextSegment = aSegment->Next(); SCH_LINE* segment = aSegments.first;
SCH_LINE* next_segment = aSegments.second;
wxPoint midPoint; wxPoint midPoint;
int iDx = aSegment->GetEndPoint().x - aSegment->GetStartPoint().x; int iDx = segment->GetEndPoint().x - segment->GetStartPoint().x;
int iDy = aSegment->GetEndPoint().y - aSegment->GetStartPoint().y; int iDy = segment->GetEndPoint().y - segment->GetStartPoint().y;
const SCH_SHEET_PIN* connectedPin = getSheetPin( aScreen, aSegment->GetStartPoint() ); const SCH_SHEET_PIN* connectedPin = getSheetPin( aScreen, segment->GetStartPoint() );
auto force = connectedPin ? connectedPin->GetEdge() : SHEET_UNDEFINED_SIDE; auto force = connectedPin ? connectedPin->GetEdge() : SHEET_UNDEFINED_SIDE;
if( force == SHEET_LEFT_SIDE || force == SHEET_RIGHT_SIDE ) if( force == SHEET_LEFT_SIDE || force == SHEET_RIGHT_SIDE )
@ -408,43 +442,44 @@ static void computeBreakPoint( SCH_SCREEN* aScreen, SCH_LINE* aSegment, wxPoint&
} }
midPoint.x = aPosition.x; midPoint.x = aPosition.x;
midPoint.y = aSegment->GetStartPoint().y; // force horizontal midPoint.y = segment->GetStartPoint().y; // force horizontal
} }
else if( iDy != 0 ) // keep the first segment orientation (vertical) else if( iDy != 0 ) // keep the first segment orientation (vertical)
{ {
midPoint.x = aSegment->GetStartPoint().x; midPoint.x = segment->GetStartPoint().x;
midPoint.y = aPosition.y; midPoint.y = aPosition.y;
} }
else if( iDx != 0 ) // keep the first segment orientation (horizontal) else if( iDx != 0 ) // keep the first segment orientation (horizontal)
{ {
midPoint.x = aPosition.x; midPoint.x = aPosition.x;
midPoint.y = aSegment->GetStartPoint().y; midPoint.y = segment->GetStartPoint().y;
} }
else else
{ {
if( std::abs( aPosition.x - aSegment->GetStartPoint().x ) < if( std::abs( aPosition.x - segment->GetStartPoint().x ) <
std::abs( aPosition.y - aSegment->GetStartPoint().y ) ) std::abs( aPosition.y - segment->GetStartPoint().y ) )
{ {
midPoint.x = aSegment->GetStartPoint().x; midPoint.x = segment->GetStartPoint().x;
midPoint.y = aPosition.y; midPoint.y = aPosition.y;
} }
else else
{ {
midPoint.x = aPosition.x; midPoint.x = aPosition.x;
midPoint.y = aSegment->GetStartPoint().y; midPoint.y = segment->GetStartPoint().y;
} }
} }
aSegment->SetEndPoint( midPoint ); segment->SetEndPoint( midPoint );
nextSegment->SetStartPoint( midPoint ); next_segment->SetStartPoint( midPoint );
nextSegment->SetEndPoint( aPosition ); next_segment->SetEndPoint( aPosition );
} }
int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType, SCH_LINE* aSegment ) int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType )
{ {
SCH_SCREEN* screen = m_frame->GetScreen(); SCH_SCREEN* screen = m_frame->GetScreen();
EE_POINT_EDITOR* pointEditor = m_toolMgr->GetTool<EE_POINT_EDITOR>(); EE_POINT_EDITOR* pointEditor = m_toolMgr->GetTool<EE_POINT_EDITOR>();
SCH_LINE* segment = nullptr;
m_toolMgr->RunAction( EE_ACTIONS::clearSelection, true ); m_toolMgr->RunAction( EE_ACTIONS::clearSelection, true );
getViewControls()->ShowCursor( true ); getViewControls()->ShowCursor( true );
@ -470,8 +505,11 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
auto cleanup = [&] () { auto cleanup = [&] () {
m_toolMgr->RunAction( EE_ACTIONS::clearSelection, true ); m_toolMgr->RunAction( EE_ACTIONS::clearSelection, true );
aSegment = nullptr; for( auto wire : m_wires )
s_wires.DeleteAll(); delete wire;
m_wires.clear();
segment = nullptr;
if( m_busUnfold.entry ) if( m_busUnfold.entry )
m_frame->RemoveFromScreen( m_busUnfold.entry ); m_frame->RemoveFromScreen( m_busUnfold.entry );
@ -492,7 +530,7 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
if( evt->IsCancelInteractive() ) if( evt->IsCancelInteractive() )
{ {
if( aSegment || m_busUnfold.in_progress ) if( segment || m_busUnfold.in_progress )
cleanup(); cleanup();
else else
{ {
@ -502,7 +540,7 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
} }
else if( evt->IsActivate() ) else if( evt->IsActivate() )
{ {
if( aSegment || m_busUnfold.in_progress ) if( segment || m_busUnfold.in_progress )
cleanup(); cleanup();
if( evt->IsMoveTool() ) if( evt->IsMoveTool() )
@ -524,16 +562,16 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
|| evt->IsAction( &EE_ACTIONS::finishBus ) || evt->IsAction( &EE_ACTIONS::finishBus )
|| evt->IsAction( &EE_ACTIONS::finishLine ) ) || evt->IsAction( &EE_ACTIONS::finishLine ) )
{ {
if( aSegment || m_busUnfold.in_progress ) if( segment || m_busUnfold.in_progress )
{ {
finishSegments(); finishSegments();
aSegment = nullptr; segment = nullptr;
} }
} }
//------------------------------------------------------------------------ //------------------------------------------------------------------------
// Handle click: // Handle click:
// //
else if( evt->IsClick( BUT_LEFT ) || ( aSegment && evt->IsDblClick( BUT_LEFT ) ) ) else if( evt->IsClick( BUT_LEFT ) || ( segment && evt->IsDblClick( BUT_LEFT ) ) )
{ {
// First click when unfolding places the label and wire-to-bus entry // First click when unfolding places the label and wire-to-bus entry
if( m_busUnfold.in_progress && !m_busUnfold.label_placed ) if( m_busUnfold.in_progress && !m_busUnfold.label_placed )
@ -545,42 +583,42 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
m_busUnfold.label_placed = true; m_busUnfold.label_placed = true;
} }
if( !aSegment ) if( !segment )
{ {
aSegment = startSegments( aType, cursorPos ); segment = startSegments( aType, VECTOR2D( cursorPos ) );
} }
// Create a new segment if we're out of previously-created ones // Create a new segment if we're out of previously-created ones
else if( !aSegment->IsNull() || ( forceHV && !aSegment->Back()->IsNull() ) ) else if( !segment->IsNull() || ( forceHV && !m_wires.end()[-2]->IsNull() ) )
{ {
// Terminate the command if the end point is on a pin, junction, or another // Terminate the command if the end point is on a pin, junction, or another
// wire or bus. // wire or bus.
if( !m_busUnfold.in_progress if( !m_busUnfold.in_progress
&& screen->IsTerminalPoint( cursorPos, aSegment->GetLayer() ) ) && screen->IsTerminalPoint( cursorPos, segment->GetLayer() ) )
{ {
finishSegments(); finishSegments();
aSegment = nullptr; segment = nullptr;
} }
else else
{ {
aSegment->SetEndPoint( cursorPos ); segment->SetEndPoint( cursorPos );
// Create a new segment, and chain it after the current segment. // Create a new segment, and chain it after the current segment.
aSegment = new SCH_LINE( *aSegment ); segment = new SCH_LINE( *segment );
aSegment->SetFlags( IS_NEW | IS_MOVED ); segment->SetFlags( IS_NEW | IS_MOVED );
aSegment->SetStartPoint( cursorPos ); segment->SetStartPoint( cursorPos );
s_wires.PushBack( aSegment ); m_wires.push_back( segment );
m_selectionTool->AddItemToSel( aSegment, true /*quiet mode*/ ); m_selectionTool->AddItemToSel( segment, true /*quiet mode*/ );
} }
} }
if( evt->IsDblClick( BUT_LEFT ) && aSegment ) if( evt->IsDblClick( BUT_LEFT ) && segment )
{ {
if( forceHV ) if( forceHV && m_wires.size() >= 2 )
computeBreakPoint( screen, aSegment->Back(), cursorPos ); computeBreakPoint( screen, { m_wires.end()[-2], segment }, cursorPos );
finishSegments(); finishSegments();
aSegment = nullptr; segment = nullptr;
} }
} }
//------------------------------------------------------------------------ //------------------------------------------------------------------------
@ -615,7 +653,7 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
m_frame->RefreshItem( entry ); m_frame->RefreshItem( entry );
wxPoint wire_start = offset ? entry->GetPosition() : entry->m_End(); wxPoint wire_start = offset ? entry->GetPosition() : entry->m_End();
s_wires.begin()->SetStartPoint( wire_start ); m_wires.front()->SetStartPoint( wire_start );
} }
// Update the label "ghost" position // Update the label "ghost" position
@ -623,19 +661,19 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
m_view->AddToPreview( m_busUnfold.label->Clone() ); m_view->AddToPreview( m_busUnfold.label->Clone() );
} }
if( aSegment ) if( segment )
{ {
// Coerce the line to vertical or horizontal if necessary // Coerce the line to vertical or horizontal if necessary
if( forceHV ) if( forceHV && m_wires.size() >= 2 )
computeBreakPoint( screen, aSegment->Back(), cursorPos ); computeBreakPoint( screen, { m_wires.end()[-2], segment }, cursorPos );
else else
aSegment->SetEndPoint( cursorPos ); segment->SetEndPoint( cursorPos );
} }
for( auto seg = s_wires.begin(); seg; seg = seg->Next() ) for( auto wire : m_wires )
{ {
if( !seg->IsNull() ) // Add to preview if segment length != 0 if( !wire->IsNull() )
m_view->AddToPreview( seg->Clone() ); m_view->AddToPreview( wire->Clone() );
} }
} }
//------------------------------------------------------------------------ //------------------------------------------------------------------------
@ -644,7 +682,7 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
else if( evt->IsClick( BUT_RIGHT ) ) else if( evt->IsClick( BUT_RIGHT ) )
{ {
// Warp after context menu only if dragging... // Warp after context menu only if dragging...
if( !aSegment ) if( !segment )
m_toolMgr->VetoContextMenuMouseWarp(); m_toolMgr->VetoContextMenuMouseWarp();
m_menu.ShowContextMenu( m_selectionTool->GetSelection() ); m_menu.ShowContextMenu( m_selectionTool->GetSelection() );
@ -654,19 +692,19 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
if( evt->GetCommandId().get() >= ID_POPUP_SCH_UNFOLD_BUS if( evt->GetCommandId().get() >= ID_POPUP_SCH_UNFOLD_BUS
&& evt->GetCommandId().get() <= ID_POPUP_SCH_UNFOLD_BUS_END ) && evt->GetCommandId().get() <= ID_POPUP_SCH_UNFOLD_BUS_END )
{ {
wxASSERT_MSG( !aSegment, "Bus unfold event received when already drawing!" ); wxASSERT_MSG( !segment, "Bus unfold event received when already drawing!" );
aType = LAYER_WIRE; aType = LAYER_WIRE;
wxString net = *evt->Parameter<wxString*>(); wxString net = *evt->Parameter<wxString*>();
aSegment = doUnfoldBus( net ); segment = doUnfoldBus( net );
} }
} }
else else
evt->SetPassEvent(); evt->SetPassEvent();
// Enable autopanning and cursor capture only when there is a segment to be placed // Enable autopanning and cursor capture only when there is a segment to be placed
getViewControls()->SetAutoPan( aSegment != nullptr ); getViewControls()->SetAutoPan( segment != nullptr );
getViewControls()->CaptureCursor( aSegment != nullptr ); getViewControls()->CaptureCursor( segment != nullptr );
} }
return 0; return 0;
@ -676,27 +714,32 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType,
SCH_LINE* SCH_LINE_WIRE_BUS_TOOL::startSegments( int aType, const VECTOR2D& aPos ) SCH_LINE* SCH_LINE_WIRE_BUS_TOOL::startSegments( int aType, const VECTOR2D& aPos )
{ {
SCH_LINE* segment = nullptr; SCH_LINE* segment = nullptr;
bool forceHV = m_frame->GetForceHVLines();
switch( aType ) switch ( aType )
{ {
default: segment = new SCH_LINE( (wxPoint) aPos, LAYER_NOTES ); break; default:
case LAYER_WIRE: segment = new SCH_LINE( (wxPoint) aPos, LAYER_WIRE ); break; segment = new SCH_LINE( aPos, LAYER_NOTES );
case LAYER_BUS: segment = new SCH_LINE( (wxPoint) aPos, LAYER_BUS ); break; break;
case LAYER_WIRE:
segment = new SCH_LINE( aPos, LAYER_WIRE );
break;
case LAYER_BUS:
segment = new SCH_LINE( aPos, LAYER_BUS );
break;
} }
segment->SetFlags( IS_NEW | IS_MOVED ); segment->SetFlags( IS_NEW | IS_MOVED );
s_wires.PushBack( segment ); m_wires.push_back( segment );
m_selectionTool->AddItemToSel( segment, true /*quiet mode*/ ); m_selectionTool->AddItemToSel( segment, true /*quiet mode*/ );
// We need 2 segments to go from a given start pin to an end point when the // We need 2 segments to go from a given start pin to an end point when the
// horizontal and vertical lines only switch is on. // horizontal and vertical lines only switch is on.
if( forceHV ) if( m_frame->GetForceHVLines() )
{ {
segment = new SCH_LINE( *segment ); segment = new SCH_LINE( *segment );
segment->SetFlags( IS_NEW | IS_MOVED ); segment->SetFlags( IS_NEW | IS_MOVED );
s_wires.PushBack( segment ); m_wires.push_back( segment );
m_selectionTool->AddItemToSel( segment, true /*quiet mode*/ ); m_selectionTool->AddItemToSel( segment, true /*quiet mode*/ );
} }
@ -718,52 +761,39 @@ SCH_LINE* SCH_LINE_WIRE_BUS_TOOL::startSegments( int aType, const VECTOR2D& aPos
* RemoveBacktracks is called: * RemoveBacktracks is called:
* -------------------> * ------------------->
*/ */
static void removeBacktracks( DLIST<SCH_LINE>& aWires ) static void removeBacktracks( std::deque<SCH_LINE*>& aWires )
{ {
SCH_LINE* next = nullptr; for( auto it = aWires.begin(); it != aWires.end(); )
std::vector<SCH_LINE*> last_lines;
for( SCH_LINE* line = aWires.GetFirst(); line; line = next )
{ {
next = line->Next(); SCH_LINE* line = *it;
if( line->IsNull() ) if( line->IsNull() )
{ {
delete s_wires.Remove( line ); delete line;
it = aWires.erase( it );
continue; continue;
} }
if( !last_lines.empty() ) auto next_it = it;
{ ++next_it;
SCH_LINE* last_line = last_lines[last_lines.size() - 1];
bool contiguous = ( last_line->GetEndPoint() == line->GetStartPoint() );
bool backtracks = IsPointOnSegment( last_line->GetStartPoint(),
last_line->GetEndPoint(), line->GetEndPoint() );
bool total_backtrack = ( last_line->GetStartPoint() == line->GetEndPoint() );
if( contiguous && backtracks ) if( next_it == aWires.end() )
{ break;
if( total_backtrack )
{ SCH_LINE* next_line = *next_it;
delete s_wires.Remove( last_line );
delete s_wires.Remove( line ); if( line->IsParallel( next_line ) )
last_lines.pop_back();
}
else
{
last_line->SetEndPoint( line->GetEndPoint() );
delete s_wires.Remove( line );
}
}
else
{
last_lines.push_back( line );
}
}
else
{ {
last_lines.push_back( line ); if( SCH_LINE* merged = line->MergeOverlap( next_line ) )
{
delete line;
delete next_line;
it = aWires.erase( it );
*it = merged;
}
} }
++it;
} }
} }
@ -778,7 +808,7 @@ void SCH_LINE_WIRE_BUS_TOOL::finishSegments()
PICKED_ITEMS_LIST itemList; PICKED_ITEMS_LIST itemList;
// Remove segments backtracking over others // Remove segments backtracking over others
removeBacktracks( s_wires ); removeBacktracks( m_wires );
// Collect the possible connection points for the new lines // Collect the possible connection points for the new lines
std::vector< wxPoint > connections; std::vector< wxPoint > connections;
@ -786,7 +816,7 @@ void SCH_LINE_WIRE_BUS_TOOL::finishSegments()
m_frame->GetSchematicConnections( connections ); m_frame->GetSchematicConnections( connections );
// Check each new segment for possible junctions and add/split if needed // Check each new segment for possible junctions and add/split if needed
for( SCH_LINE* wire = s_wires.GetFirst(); wire; wire = wire->Next() ) for( auto wire : m_wires )
{ {
if( wire->HasFlag( SKIP_STRUCT ) ) if( wire->HasFlag( SKIP_STRUCT ) )
continue; continue;
@ -811,15 +841,17 @@ void SCH_LINE_WIRE_BUS_TOOL::finishSegments()
} }
// Get the last non-null wire (this is the last created segment). // Get the last non-null wire (this is the last created segment).
m_frame->SaveCopyForRepeatItem( s_wires.GetLast() ); if( !m_wires.empty() )
m_frame->SaveCopyForRepeatItem( m_wires.back() );
// Add the new wires // Add the new wires
while( s_wires.GetFirst() ) for( auto wire : m_wires )
{ {
s_wires.GetFirst()->ClearFlags( IS_NEW | IS_MOVED ); wire->ClearFlags( IS_NEW | IS_MOVED );
m_frame->AddToScreen( s_wires.PopFront() ); m_frame->AddToScreen( wire );
} }
m_wires.clear();
m_view->ClearPreview(); m_view->ClearPreview();
m_view->ShowPreview( false ); m_view->ShowPreview( false );

View File

@ -81,7 +81,7 @@ public:
int AddJunctionsIfNeeded( const TOOL_EVENT& aEvent ); int AddJunctionsIfNeeded( const TOOL_EVENT& aEvent );
private: private:
int doDrawSegments( const std::string& aTool, int aType, SCH_LINE* aSegment ); int doDrawSegments( const std::string& aTool, int aType );
SCH_LINE* startSegments( int aType, const VECTOR2D& aPos ); SCH_LINE* startSegments( int aType, const VECTOR2D& aPos );
SCH_LINE* doUnfoldBus( const wxString& aNet ); SCH_LINE* doUnfoldBus( const wxString& aNet );
void finishSegments(); void finishSegments();
@ -91,7 +91,10 @@ private:
private: private:
/// Data related to bus unfolding tool. /// Data related to bus unfolding tool.
BUS_UNFOLDING_T m_busUnfold; BUS_UNFOLDING_T m_busUnfold;
/// Storage for the line segments while drawing
std::deque<SCH_LINE*> m_wires;
}; };
#endif /* SCH_LINE_WIRE_BUS_TOOL_H */ #endif /* SCH_LINE_WIRE_BUS_TOOL_H */