pcbnew: Improve zone/polygon selection
Allows selection of polygons and zones by their visible area. Will only
select polygons/filled areas when they are unobstructed by other
elements in the footprint/pcb. Also adds a "skip heuristics" modifier
key for clicking (ALT) that allows the full disambiguation list to be
displayed without removing less-likely elements. This dovetails the
zone/polygon selection as it allows selecting even when areas a fully
overlapped.
Fixes: lp:1753153
* https://bugs.launchpad.net/kicad/+bug/1753153
(cherry picked from commit 6a6d580a1c
)
This commit is contained in:
parent
6994291997
commit
5e0ef36719
|
@ -591,14 +591,9 @@ bool DRAWSEGMENT::HitTest( const wxPoint& aPosition ) const
|
||||||
case S_POLYGON: // not yet handled
|
case S_POLYGON: // not yet handled
|
||||||
{
|
{
|
||||||
#define MAX_DIST_IN_MM 0.25
|
#define MAX_DIST_IN_MM 0.25
|
||||||
int distmax = Millimeter2iu( 0.25 );
|
int distmax = std::max( m_Width, Millimeter2iu( MAX_DIST_IN_MM ) );
|
||||||
SHAPE_POLY_SET::VERTEX_INDEX dummy;
|
|
||||||
auto poly = m_Poly;
|
|
||||||
|
|
||||||
if( poly.CollideVertex( VECTOR2I( aPosition ), dummy, distmax ) )
|
if( m_Poly.Collide( VECTOR2I( aPosition ), distmax ) )
|
||||||
return true;
|
|
||||||
|
|
||||||
if( poly.CollideEdge( VECTOR2I( aPosition ), dummy, distmax ) )
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -653,7 +653,8 @@ void ZONE_CONTAINER::SetCornerRadius( unsigned int aRadius )
|
||||||
|
|
||||||
bool ZONE_CONTAINER::HitTest( const wxPoint& aPosition ) const
|
bool ZONE_CONTAINER::HitTest( const wxPoint& aPosition ) const
|
||||||
{
|
{
|
||||||
return HitTestForCorner( aPosition ) || HitTestForEdge( aPosition );
|
return HitTestForCorner( aPosition ) || HitTestForEdge( aPosition ) ||
|
||||||
|
HitTestFilledArea( aPosition );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -191,6 +191,7 @@ SELECTION_TOOL::SELECTION_TOOL() :
|
||||||
m_additive( false ),
|
m_additive( false ),
|
||||||
m_subtractive( false ),
|
m_subtractive( false ),
|
||||||
m_multiple( false ),
|
m_multiple( false ),
|
||||||
|
m_skip_heuristics( false ),
|
||||||
m_locked( true ),
|
m_locked( true ),
|
||||||
m_menu( *this ),
|
m_menu( *this ),
|
||||||
m_priv( std::make_unique<PRIV>() )
|
m_priv( std::make_unique<PRIV>() )
|
||||||
|
@ -270,6 +271,10 @@ int SELECTION_TOOL::Main( const TOOL_EVENT& aEvent )
|
||||||
// This will be ignored if the SHIFT modifier is pressed
|
// This will be ignored if the SHIFT modifier is pressed
|
||||||
m_subtractive = !m_additive && evt->Modifier( MD_CTRL );
|
m_subtractive = !m_additive && evt->Modifier( MD_CTRL );
|
||||||
|
|
||||||
|
// Is the user requesting that the selection list include all possible
|
||||||
|
// items without removing less likely selection candidates
|
||||||
|
m_skip_heuristics = !!evt->Modifier( MD_ALT );
|
||||||
|
|
||||||
// Single click? Select single object
|
// Single click? Select single object
|
||||||
if( evt->IsClick( BUT_LEFT ) )
|
if( evt->IsClick( BUT_LEFT ) )
|
||||||
{
|
{
|
||||||
|
@ -509,12 +514,15 @@ bool SELECTION_TOOL::selectPoint( const VECTOR2I& aWhere, bool aOnDrag,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Apply some ugly heuristics to avoid disambiguation menus whenever possible
|
// Apply some ugly heuristics to avoid disambiguation menus whenever possible
|
||||||
guessSelectionCandidates( collector );
|
if( !m_skip_heuristics )
|
||||||
|
|
||||||
if( collector.GetCount() == 1 )
|
|
||||||
{
|
{
|
||||||
toggleSelection( collector[0] );
|
guessSelectionCandidates( collector );
|
||||||
return true;
|
|
||||||
|
if( collector.GetCount() == 1 )
|
||||||
|
{
|
||||||
|
toggleSelection( collector[0] );
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Still more than one item. We're going to have to ask the user.
|
// Still more than one item. We're going to have to ask the user.
|
||||||
|
@ -1990,6 +1998,30 @@ void SELECTION_TOOL::guessSelectionCandidates( GENERAL_COLLECTOR& aCollector ) c
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int numZones = aCollector.CountType( PCB_ZONE_AREA_T );
|
||||||
|
|
||||||
|
if( numZones > 0 && aCollector.GetCount() > numZones )
|
||||||
|
{
|
||||||
|
for( int i = aCollector.GetCount() - 1; i >= 0; i-- )
|
||||||
|
{
|
||||||
|
if( aCollector[i]->Type() == PCB_ZONE_AREA_T )
|
||||||
|
aCollector.Remove( i );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int numDrawitems = aCollector.CountType( PCB_LINE_T ) +
|
||||||
|
aCollector.CountType( PCB_MODULE_EDGE_T );
|
||||||
|
|
||||||
|
if( numDrawitems > 0 && aCollector.GetCount() > numDrawitems )
|
||||||
|
{
|
||||||
|
for( int i = aCollector.GetCount() - 1; i >= 0; i-- )
|
||||||
|
{
|
||||||
|
auto ds = static_cast<DRAWSEGMENT*>( aCollector[i] );
|
||||||
|
if( ds->GetShape() == S_POLYGON )
|
||||||
|
aCollector.Remove( i );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if( aCollector.CountType( PCB_MODULE_TEXT_T ) > 0 )
|
if( aCollector.CountType( PCB_MODULE_TEXT_T ) > 0 )
|
||||||
{
|
{
|
||||||
for( int i = 0; i < aCollector.GetCount(); ++i )
|
for( int i = 0; i < aCollector.GetCount(); ++i )
|
||||||
|
|
|
@ -351,6 +351,9 @@ private:
|
||||||
/// Flag saying if multiple selection mode is active.
|
/// Flag saying if multiple selection mode is active.
|
||||||
bool m_multiple;
|
bool m_multiple;
|
||||||
|
|
||||||
|
/// Flag saying that heuristics should be skipped while choosing selection
|
||||||
|
bool m_skip_heuristics;
|
||||||
|
|
||||||
/// Can other tools modify locked items.
|
/// Can other tools modify locked items.
|
||||||
bool m_locked;
|
bool m_locked;
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue