KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcb_selection_tool.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2013-2017 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
7 * @author Maciej Suminski <maciej.suminski@cern.ch>
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, you may find one here:
21 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
22 * or you may search the http://www.gnu.org website for the version 2 license,
23 * or you may write to the Free Software Foundation, Inc.,
24 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
25 */
26
27#include <limits>
28#include <cmath>
29#include <functional>
30#include <stack>
31using namespace std::placeholders;
32
33#include <advanced_config.h>
34#include <macros.h>
35#include <core/kicad_algo.h>
36#include <board.h>
38#include <board_item.h>
39#include <pcb_reference_image.h>
40#include <pcb_track.h>
41#include <footprint.h>
42#include <pad.h>
43#include <pcb_group.h>
44#include <pcb_shape.h>
45#include <pcb_text.h>
46#include <pcb_textbox.h>
47#include <pcb_table.h>
48#include <pcb_tablecell.h>
49#include <pcb_marker.h>
50#include <pcb_generator.h>
51#include <zone.h>
52#include <collectors.h>
56#include <view/view_controls.h>
58#include <gal/painter.h>
59#include <router/router_tool.h>
60#include <pcbnew_settings.h>
61#include <tool/tool_event.h>
62#include <tool/tool_manager.h>
66#include <tools/pcb_actions.h>
71#include <wx/event.h>
72#include <wx/timer.h>
73#include <wx/log.h>
74#include <wx/debug.h>
75#include <core/profile.h>
76#include <math/vector2wx.h>
77
78
80{
82 double m_Opacity;
84};
85
86
88{
89public:
91 ACTION_MENU( true )
92 {
93 SetTitle( _( "Select" ) );
94
96
97 AppendSeparator();
98
101
102 // This could be enabled if we have better logic for picking the target net with the mouse
103 // Add( PCB_ACTIONS::deselectNet );
106
109 }
110
111private:
112 ACTION_MENU* create() const override
113 {
114 return new SELECT_MENU();
115 }
116};
117
118
123{
124public:
126};
127
128
130 SELECTION_TOOL( "common.InteractiveSelection" ),
131 m_frame( nullptr ),
132 m_isFootprintEditor( false ),
133 m_nonModifiedCursor( KICURSOR::ARROW ),
134 m_enteredGroup( nullptr ),
135 m_priv( std::make_unique<PRIV>() )
136{
137 m_filter.lockedItems = false;
138 m_filter.footprints = true;
139 m_filter.text = true;
140 m_filter.tracks = true;
141 m_filter.vias = true;
142 m_filter.pads = true;
143 m_filter.graphics = true;
144 m_filter.zones = true;
145 m_filter.keepouts = true;
146 m_filter.dimensions = true;
147 m_filter.otherItems = true;
148}
149
150
152{
155
156 Disconnect( wxEVT_TIMER, wxTimerEventHandler( PCB_SELECTION_TOOL::onDisambiguationExpire ),
157 nullptr, this );
158}
159
160
162{
163 PCB_BASE_FRAME* frame = getEditFrame<PCB_BASE_FRAME>();
164
166 {
168 return true;
169 }
170
171 std::shared_ptr<SELECT_MENU> selectMenu = std::make_shared<SELECT_MENU>();
172 selectMenu->SetTool( this );
173 m_menu->RegisterSubMenu( selectMenu );
174
175 static const std::vector<KICAD_T> tableCellTypes = { PCB_TABLECELL_T };
176
177 auto& menu = m_menu->GetMenu();
178
179 auto activeToolCondition =
180 [ frame ] ( const SELECTION& aSel )
181 {
182 return !frame->ToolStackIsEmpty();
183 };
184
185 auto haveHighlight =
186 [&]( const SELECTION& sel )
187 {
189
190 return !cfg->GetHighlightNetCodes().empty();
191 };
192
193 auto groupEnterCondition =
195
196 auto inGroupCondition =
197 [this] ( const SELECTION& )
198 {
199 return m_enteredGroup != nullptr;
200 };
201
202 auto tableCellSelection = SELECTION_CONDITIONS::MoreThan( 0 )
204
206 {
207 menu.AddMenu( selectMenu.get(), SELECTION_CONDITIONS::NotEmpty );
208 menu.AddSeparator( 1000 );
209 }
210
211 // "Cancel" goes at the top of the context menu when a tool is active
212 menu.AddItem( ACTIONS::cancelInteractive, activeToolCondition, 1 );
213 menu.AddItem( ACTIONS::groupEnter, groupEnterCondition, 1 );
214 menu.AddItem( ACTIONS::groupLeave, inGroupCondition, 1 );
215 menu.AddItem( PCB_ACTIONS::clearHighlight, haveHighlight, 1 );
216 menu.AddSeparator( haveHighlight, 1 );
217
218 menu.AddItem( ACTIONS::selectColumns, tableCellSelection, 2 );
219 menu.AddItem( ACTIONS::selectRows, tableCellSelection, 2 );
220 menu.AddItem( ACTIONS::selectTable, tableCellSelection, 2 );
221
222 menu.AddSeparator( 1 );
223
224 if( frame )
226
227 m_disambiguateTimer.SetOwner( this );
228 Connect( wxEVT_TIMER, wxTimerEventHandler( PCB_SELECTION_TOOL::onDisambiguationExpire ),
229 nullptr, this );
230
231 return true;
232}
233
234
236{
237 m_frame = getEditFrame<PCB_BASE_FRAME>();
239
240 if( aReason != TOOL_BASE::REDRAW )
241 {
242 if( m_enteredGroup )
243 ExitGroup();
244
245 // Deselect any item being currently in edit, to avoid unexpected behavior and remove
246 // pointers to the selected items from containers.
247 ClearSelection( true );
248 }
249
250 if( aReason == TOOL_BASE::MODEL_RELOAD )
251 getView()->GetPainter()->GetSettings()->SetHighlight( false );
252
253 // Reinsert the VIEW_GROUP, in case it was removed from the VIEW
254 view()->Remove( &m_selection );
255 view()->Add( &m_selection );
256
259}
260
261
262void PCB_SELECTION_TOOL::OnIdle( wxIdleEvent& aEvent )
263{
265 {
266 wxMouseState keyboardState = wxGetMouseState();
267
268 setModifiersState( keyboardState.ShiftDown(), keyboardState.ControlDown(),
269 keyboardState.AltDown() );
270
271 if( m_additive )
272 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ADD );
273 else if( m_subtractive )
274 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::SUBTRACT );
275 else if( m_exclusive_or )
276 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::XOR );
277 else
279 }
280}
281
282
284{
285 // Main loop: keep receiving events
286 while( TOOL_EVENT* evt = Wait() )
287 {
289 TRACK_DRAG_ACTION trackDragAction = TRACK_DRAG_ACTION::MOVE;
290
291 try
292 {
293 trackDragAction = m_frame->GetPcbNewSettings()->m_TrackDragAction;
294 }
295 catch( const std::runtime_error& e )
296 {
297 wxFAIL_MSG( e.what() );
298 }
299
300 // on left click, a selection is made, depending on modifiers ALT, SHIFT, CTRL:
301 setModifiersState( evt->Modifier( MD_SHIFT ), evt->Modifier( MD_CTRL ),
302 evt->Modifier( MD_ALT ) );
303
304 PCB_BASE_FRAME* frame = getEditFrame<PCB_BASE_FRAME>();
305 bool brd_editor = frame && frame->IsType( FRAME_PCB_EDITOR );
307
308 // If the router tool is active, don't override
309 if( router && router->IsToolActive() && router->RoutingInProgress() )
310 {
311 evt->SetPassEvent();
312 }
313 else if( evt->IsMouseDown( BUT_LEFT ) )
314 {
315 // Avoid triggering when running under other tools
317
318 if( m_frame->ToolStackIsEmpty() && pt_tool && !pt_tool->HasPoint() )
319 {
322 }
323 }
324 else if( evt->IsClick( BUT_LEFT ) )
325 {
326 // If there is no disambiguation, this routine is still running and will
327 // register a `click` event when released
328 if( m_disambiguateTimer.IsRunning() )
329 {
330 m_disambiguateTimer.Stop();
331
332 // Single click? Select single object
333 if( m_highlight_modifier && brd_editor )
334 {
336 }
337 else
338 {
340 selectPoint( evt->Position() );
341 }
342 }
343
344 m_canceledMenu = false;
345 }
346 else if( evt->IsClick( BUT_RIGHT ) )
347 {
348 m_disambiguateTimer.Stop();
349
350 // Right click? if there is any object - show the context menu
351 bool selectionCancelled = false;
352
353 if( m_selection.Empty() )
354 {
355 selectPoint( evt->Position(), false, &selectionCancelled );
356 m_selection.SetIsHover( true );
357 }
358
359 // Show selection before opening menu
361
362 if( !selectionCancelled )
363 {
365 m_menu->ShowContextMenu( m_selection );
366 }
367 }
368 else if( evt->IsDblClick( BUT_LEFT ) )
369 {
370 m_disambiguateTimer.Stop();
371
372 // Double clicks make no sense in the footprint viewer
374 {
375 evt->SetPassEvent();
376 continue;
377 }
378
379 // Double click? Display the properties window
381
382 if( m_selection.Empty() )
383 selectPoint( evt->Position() );
384
385 if( m_selection.GetSize() == 1 && m_selection[0]->Type() == PCB_GROUP_T )
386 EnterGroup();
387 else
389 }
390 else if( evt->IsDblClick( BUT_MIDDLE ) )
391 {
392 // Middle double click? Do zoom to fit or zoom to objects
393 if( evt->Modifier( MD_CTRL ) ) // Is CTRL key down?
395 else
397 }
398 else if( evt->Action() == TA_MOUSE_WHEEL )
399 {
400 int field = -1;
401
402 if( evt->Modifier() == ( MD_SHIFT | MD_ALT ) )
403 field = 0;
404 else if( evt->Modifier() == ( MD_CTRL | MD_ALT ) )
405 field = 1;
406 // any more?
407
408 if( field >= 0 )
409 {
410 const int delta = evt->Parameter<int>();
411 ACTIONS::INCREMENT incParams{
412 delta > 0 ? 1 : -1,
413 field,
414 };
415
417 }
418 }
419 else if( evt->IsDrag( BUT_LEFT ) )
420 {
421 m_disambiguateTimer.Stop();
422
423 // Is another tool already moving a new object? Don't allow a drag start
424 if( !m_selection.Empty() && m_selection[0]->HasFlag( IS_NEW | IS_MOVING ) )
425 {
426 evt->SetPassEvent();
427 continue;
428 }
429
430 // Drag with LMB? Select multiple objects (or at least draw a selection box)
431 // or drag them
434
436 GENERAL_COLLECTOR collector;
437
439 {
440 if( board()->GetFirstFootprint() )
441 {
442 collector.Collect( board()->GetFirstFootprint(), { PCB_TABLECELL_T },
443 evt->DragOrigin(), guide );
444 }
445 }
446 else
447 {
448 collector.Collect( board(), { PCB_TABLECELL_T }, evt->DragOrigin(), guide );
449 }
450
451 if( collector.GetCount() )
452 {
453 if( m_selection.GetSize() == 1 && dynamic_cast<PCB_TABLE*>( m_selection.GetItem( 0 ) ) )
454 {
456 }
457 else
458 {
459 selectTableCells( static_cast<PCB_TABLE*>( collector[0]->GetParent() ) );
460 }
461 }
462 else if( hasModifier() || dragAction == MOUSE_DRAG_ACTION::SELECT )
463 {
465 }
466 else if( m_selection.Empty() && dragAction != MOUSE_DRAG_ACTION::DRAG_ANY )
467 {
469 }
470 else
471 {
472 // Don't allow starting a drag from a zone filled area that isn't already selected
473 auto zoneFilledAreaFilter =
474 []( const VECTOR2I& aWhere, GENERAL_COLLECTOR& aCollector,
475 PCB_SELECTION_TOOL* aTool )
476 {
477 int accuracy = aCollector.GetGuide()->Accuracy();
478 std::set<EDA_ITEM*> remove;
479
480 for( EDA_ITEM* item : aCollector )
481 {
482 if( item->Type() == PCB_ZONE_T )
483 {
484 ZONE* zone = static_cast<ZONE*>( item );
485
486 if( !zone->HitTestForCorner( aWhere, accuracy * 2 )
487 && !zone->HitTestForEdge( aWhere, accuracy ) )
488 {
489 remove.insert( zone );
490 }
491 }
492 }
493
494 for( EDA_ITEM* item : remove )
495 aCollector.Remove( item );
496 };
497
498 // See if we can drag before falling back to selectMultiple()
499 bool doDrag = false;
500
501 if( evt->HasPosition() )
502 {
503 if( m_selection.Empty()
504 && selectPoint( evt->DragOrigin(), false, nullptr, zoneFilledAreaFilter ) )
505 {
506 m_selection.SetIsHover( true );
507 doDrag = true;
508 }
509 // Check if dragging has started within any of selected items bounding box.
510 else if( selectionContains( evt->DragOrigin() ) )
511 {
512 doDrag = true;
513 }
514 }
515
516 if( doDrag )
517 {
518 size_t segs = m_selection.CountType( PCB_TRACE_T );
519 size_t arcs = m_selection.CountType( PCB_ARC_T );
520 size_t vias = m_selection.CountType( PCB_VIA_T );
521 // Note: multi-track dragging is currently supported, but not multi-via
522 bool routable = ( segs >= 1 || arcs >= 1 || vias == 1 )
523 && ( segs + arcs + vias == m_selection.GetSize() );
524
525 if( routable && trackDragAction == TRACK_DRAG_ACTION::DRAG )
527 else if( routable && trackDragAction == TRACK_DRAG_ACTION::DRAG_FREE_ANGLE )
529 else
531 }
532 else
533 {
534 // Otherwise drag a selection box
536 }
537 }
538 }
539 else if( evt->IsCancel() )
540 {
541 m_disambiguateTimer.Stop();
543
544 if( !GetSelection().Empty() )
545 {
547 }
548 else if( evt->FirstResponder() == this && evt->GetCommandId() == (int) WXK_ESCAPE )
549 {
550 if( m_enteredGroup )
551 {
552 ExitGroup();
553 }
554 else
555 {
557
558 try
559 {
561 controller->ClearHighlight( *evt );
562 }
563 catch( const std::runtime_error& e )
564 {
565 wxCHECK_MSG( false, 0, e.what() );
566 }
567 }
568 }
569 }
570 else
571 {
572 evt->SetPassEvent();
573 }
574
575
577 {
578 // move cursor prediction
579 if( !hasModifier()
580 && dragAction == MOUSE_DRAG_ACTION::DRAG_SELECTED
581 && !m_selection.Empty()
582 && evt->HasPosition()
583 && selectionContains( evt->Position() ) )
584 {
585 m_nonModifiedCursor = KICURSOR::MOVING;
586 }
587 else
588 {
589 m_nonModifiedCursor = KICURSOR::ARROW;
590 }
591 }
592 }
593
594 // Shutting down; clear the selection
596 m_disambiguateTimer.Stop();
597
598 return 0;
599}
600
601
603{
604 wxCHECK_RET( m_selection.GetSize() == 1 && m_selection[0]->Type() == PCB_GROUP_T,
605 wxT( "EnterGroup called when selection is not a single group" ) );
606 PCB_GROUP* aGroup = static_cast<PCB_GROUP*>( m_selection[0] );
607
608 if( m_enteredGroup != nullptr )
609 ExitGroup();
610
612 m_enteredGroup = aGroup;
615 {
616 select( titem );
617 },
618 RECURSE_MODE::NO_RECURSE );
619
621
622 view()->Hide( m_enteredGroup, true );
625}
626
627
628void PCB_SELECTION_TOOL::ExitGroup( bool aSelectGroup )
629{
630 // Only continue if there is a group entered
631 if( m_enteredGroup == nullptr )
632 return;
633
635 view()->Hide( m_enteredGroup, false );
637
638 if( aSelectGroup )
639 {
642 }
643
645 m_enteredGroup = nullptr;
647}
648
649
651{
652 return m_selection;
653}
654
655
657 bool aConfirmLockedItems )
658{
659 bool selectionEmpty = m_selection.Empty();
660 m_selection.SetIsHover( selectionEmpty );
661
662 if( selectionEmpty )
663 {
666 }
667
668 if( aClientFilter )
669 {
670 enum DISPOSITION { BEFORE = 1, AFTER, BOTH };
671
672 std::map<EDA_ITEM*, DISPOSITION> itemDispositions;
674 GENERAL_COLLECTOR collector;
675
676 collector.SetGuide( &guide );
677
678 for( EDA_ITEM* item : m_selection )
679 {
680 collector.Append( item );
681 itemDispositions[ item ] = BEFORE;
682 }
683
684 aClientFilter( VECTOR2I(), collector, this );
685
686 for( EDA_ITEM* item : collector )
687 {
688 if( itemDispositions.count( item ) )
689 itemDispositions[ item ] = BOTH;
690 else
691 itemDispositions[ item ] = AFTER;
692 }
693
694 // Unhighlight the BEFORE items before highlighting the AFTER items.
695 // This is so that in the case of groups, if aClientFilter replaces a selection
696 // with the enclosing group, the unhighlight of the element doesn't undo the
697 // recursive highlighting of that element by the group.
698
699 for( std::pair<EDA_ITEM* const, DISPOSITION> itemDisposition : itemDispositions )
700 {
701 EDA_ITEM* item = itemDisposition.first;
702 DISPOSITION disposition = itemDisposition.second;
703
704 if( disposition == BEFORE )
706 }
707
708 for( std::pair<EDA_ITEM* const, DISPOSITION> itemDisposition : itemDispositions )
709 {
710 EDA_ITEM* item = itemDisposition.first;
711 DISPOSITION disposition = itemDisposition.second;
712
713 // Note that we must re-highlight even previously-highlighted items
714 // (ie: disposition BOTH) in case we removed any of their children.
715 if( disposition == AFTER || disposition == BOTH )
716 highlight( item, SELECTED, &m_selection );
717 }
718
720 }
721
722 if( aConfirmLockedItems )
723 {
724 std::vector<BOARD_ITEM*> lockedItems;
725
726 for( EDA_ITEM* item : m_selection )
727 {
728 BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( item );
729 bool lockedDescendant = false;
730
731 boardItem->RunOnChildren(
732 [&]( BOARD_ITEM* curr_item )
733 {
734 if( curr_item->IsLocked() )
735 lockedDescendant = true;
736 },
737 RECURSE_MODE::RECURSE );
738
739 if( boardItem->IsLocked() || lockedDescendant )
740 lockedItems.push_back( boardItem );
741 }
742
744 if( !lockedItems.empty() && !settings->m_LockingOptions.m_sessionSkipPrompts )
745 {
746 DIALOG_LOCKED_ITEMS_QUERY dlg( frame(), lockedItems.size(),
747 settings->m_LockingOptions );
748
749 switch( dlg.ShowModal() )
750 {
751 case wxID_OK:
752 // remove locked items from selection
753 for( BOARD_ITEM* item : lockedItems )
754 unselect( item );
755
756 break;
757
758 case wxID_CANCEL:
759 // cancel operation
761 break;
762
763 case wxID_APPLY:
764 // continue with operation with current selection
765 break;
766 }
767 }
768 }
769
770 return m_selection;
771}
772
773
775{
776 GENERAL_COLLECTORS_GUIDE guide( board()->GetVisibleLayers(),
777 (PCB_LAYER_ID) view()->GetTopLayer(), view() );
778
779 bool padsDisabled = !board()->IsElementVisible( LAYER_PADS );
780
781 // account for the globals
782 guide.SetIgnoreFPTextOnBack( !board()->IsElementVisible( LAYER_FP_TEXT ) );
783 guide.SetIgnoreFPTextOnFront( !board()->IsElementVisible( LAYER_FP_TEXT ) );
784 guide.SetIgnoreFootprintsOnBack( !board()->IsElementVisible( LAYER_FOOTPRINTS_BK ) );
785 guide.SetIgnoreFootprintsOnFront( !board()->IsElementVisible( LAYER_FOOTPRINTS_FR ) );
786 guide.SetIgnorePadsOnBack( padsDisabled );
787 guide.SetIgnorePadsOnFront( padsDisabled );
788 guide.SetIgnoreThroughHolePads( padsDisabled );
789 guide.SetIgnoreFPValues( !board()->IsElementVisible( LAYER_FP_VALUES ) );
790 guide.SetIgnoreFPReferences( !board()->IsElementVisible( LAYER_FP_REFERENCES ) );
791 guide.SetIgnoreThroughVias( ! board()->IsElementVisible( LAYER_VIAS ) );
792 guide.SetIgnoreBlindBuriedVias( ! board()->IsElementVisible( LAYER_VIAS ) );
793 guide.SetIgnoreMicroVias( ! board()->IsElementVisible( LAYER_VIAS ) );
794 guide.SetIgnoreTracks( ! board()->IsElementVisible( LAYER_TRACKS ) );
795
796 return guide;
797}
798
799
801{
803}
804
805
806bool PCB_SELECTION_TOOL::selectPoint( const VECTOR2I& aWhere, bool aOnDrag,
807 bool* aSelectionCancelledFlag,
808 CLIENT_SELECTION_FILTER aClientFilter )
809{
811 GENERAL_COLLECTOR collector;
812 const PCB_DISPLAY_OPTIONS& displayOpts = m_frame->GetDisplayOptions();
813
814 guide.SetIgnoreZoneFills( displayOpts.m_ZoneDisplayMode != ZONE_DISPLAY_MODE::SHOW_FILLED );
815
817 ExitGroup();
818
821 aWhere, guide );
822
823 // Remove unselectable items
824 for( int i = collector.GetCount() - 1; i >= 0; --i )
825 {
826 if( !Selectable( collector[ i ] ) || ( aOnDrag && collector[i]->IsLocked() ) )
827 collector.Remove( i );
828 }
829
831
832 // Apply the stateful filter (remove items disabled by the Selection Filter)
833 FilterCollectedItems( collector, false );
834
835 // Allow the client to do tool- or action-specific filtering to see if we can get down
836 // to a single item
837 if( aClientFilter )
838 aClientFilter( aWhere, collector, this );
839
840 FilterCollectorForHierarchy( collector, false );
841
842 FilterCollectorForFootprints( collector, aWhere );
843
844 // For subtracting, we only want items that are selected
845 if( m_subtractive )
846 {
847 for( int i = collector.GetCount() - 1; i >= 0; --i )
848 {
849 if( !collector[i]->IsSelected() )
850 collector.Remove( i );
851 }
852 }
853
854 // Apply some ugly heuristics to avoid disambiguation menus whenever possible
855 if( collector.GetCount() > 1 && !m_skip_heuristics )
856 {
857 try
858 {
859 GuessSelectionCandidates( collector, aWhere );
860 }
861 catch( const std::exception& exc )
862 {
863 wxLogWarning( wxS( "Exception \"%s\" occurred attempting to guess selection "
864 "candidates." ), exc.what() );
865 return false;
866 }
867 }
868
869 // If still more than one item we're going to have to ask the user.
870 if( collector.GetCount() > 1 )
871 {
872 if( aOnDrag )
874
875 if( !doSelectionMenu( &collector ) )
876 {
877 if( aSelectionCancelledFlag )
878 *aSelectionCancelledFlag = true;
879
880 return false;
881 }
882 }
883
884 int addedCount = 0;
885 bool anySubtracted = false;
886
888 {
889 if( m_selection.GetSize() > 0 )
890 {
891 ClearSelection( true /*quiet mode*/ );
892 anySubtracted = true;
893 }
894 }
895
896 if( collector.GetCount() > 0 )
897 {
898 for( int i = 0; i < collector.GetCount(); ++i )
899 {
900 if( m_subtractive || ( m_exclusive_or && collector[i]->IsSelected() ) )
901 {
902 unselect( collector[i] );
903 anySubtracted = true;
904 }
905 else
906 {
907 select( collector[i] );
908 addedCount++;
909 }
910 }
911 }
912
913 if( addedCount == 1 )
914 {
916 return true;
917 }
918 else if( addedCount > 1 )
919 {
921 return true;
922 }
923 else if( anySubtracted )
924 {
926 return true;
927 }
928
929 return false;
930}
931
932
933bool PCB_SELECTION_TOOL::selectCursor( bool aForceSelect, CLIENT_SELECTION_FILTER aClientFilter )
934{
935 if( aForceSelect || m_selection.Empty() )
936 {
937 ClearSelection( true /*quiet mode*/ );
938 selectPoint( getViewControls()->GetCursorPosition( false ), false, nullptr, aClientFilter );
939 }
940
941 return !m_selection.Empty();
942}
943
944
945// Some navigation actions are allowed in selectMultiple
955 &ACTIONS::zoomFitObjects, nullptr };
956
957
959{
960 bool cancelled = false; // Was the tool canceled while it was running?
961 m_multiple = true; // Multiple selection mode is active
962
963 for( PCB_TABLECELL* cell : aTable->GetCells() )
964 {
965 if( cell->IsSelected() )
966 cell->SetFlags( CANDIDATE );
967 else
968 cell->ClearFlags( CANDIDATE );
969 }
970
971 auto wasSelected =
972 []( EDA_ITEM* aItem )
973 {
974 return ( aItem->GetFlags() & CANDIDATE ) > 0;
975 };
976
977 while( TOOL_EVENT* evt = Wait() )
978 {
979 if( evt->IsCancelInteractive() || evt->IsActivate() )
980 {
981 cancelled = true;
982 break;
983 }
984 else if( evt->IsDrag( BUT_LEFT ) )
985 {
986 getViewControls()->SetAutoPan( true );
987
988 BOX2I selectionRect( evt->DragOrigin(), evt->Position() - evt->DragOrigin() );
989 selectionRect.Normalize();
990
991 for( PCB_TABLECELL* cell : aTable->GetCells() )
992 {
993 bool doSelect = false;
994
995 if( cell->HitTest( selectionRect, false ) )
996 {
997 if( m_subtractive )
998 doSelect = false;
999 else if( m_exclusive_or )
1000 doSelect = !wasSelected( cell );
1001 else
1002 doSelect = true;
1003 }
1004 else if( wasSelected( cell ) )
1005 {
1006 doSelect = m_additive || m_subtractive || m_exclusive_or;
1007 }
1008
1009 if( doSelect && !cell->IsSelected() )
1010 select( cell );
1011 else if( !doSelect && cell->IsSelected() )
1012 unselect( cell );
1013 }
1014 }
1015 else if( evt->IsMouseUp( BUT_LEFT ) )
1016 {
1017 m_selection.SetIsHover( false );
1018
1019 bool anyAdded = false;
1020 bool anySubtracted = false;
1021
1022 for( PCB_TABLECELL* cell : aTable->GetCells() )
1023 {
1024 if( cell->IsSelected() && !wasSelected( cell ) )
1025 anyAdded = true;
1026 else if( wasSelected( cell ) && !cell->IsSelected() )
1027 anySubtracted = true;
1028 }
1029
1030 // Inform other potentially interested tools
1031 if( anyAdded )
1033
1034 if( anySubtracted )
1036
1037 break; // Stop waiting for events
1038 }
1039 else
1040 {
1041 // Allow some actions for navigation
1042 for( int i = 0; allowedActions[i]; ++i )
1043 {
1044 if( evt->IsAction( allowedActions[i] ) )
1045 {
1046 evt->SetPassEvent();
1047 break;
1048 }
1049 }
1050 }
1051 }
1052
1053 getViewControls()->SetAutoPan( false );
1054
1055 m_multiple = false; // Multiple selection mode is inactive
1056
1057 if( !cancelled )
1059
1060 return cancelled;
1061}
1062
1063
1065{
1066 bool cancelled = false; // Was the tool canceled while it was running?
1067 m_multiple = true; // Multiple selection mode is active
1069
1071 view->Add( &area );
1072
1073 bool anyAdded = false;
1074 bool anySubtracted = false;
1075
1076 while( TOOL_EVENT* evt = Wait() )
1077 {
1078 int width = area.GetEnd().x - area.GetOrigin().x;
1079
1080 /* Selection mode depends on direction of drag-selection:
1081 * Left > Right : Select objects that are fully enclosed by selection
1082 * Right > Left : Select objects that are crossed by selection
1083 */
1084 bool greedySelection = width >= 0 ? false : true;
1085
1086 if( view->IsMirroredX() )
1087 greedySelection = !greedySelection;
1088
1089 m_frame->GetCanvas()->SetCurrentCursor( !greedySelection ? KICURSOR::SELECT_WINDOW
1090 : KICURSOR::SELECT_LASSO );
1091
1092 if( evt->IsCancelInteractive() || evt->IsActivate() )
1093 {
1094 cancelled = true;
1095 break;
1096 }
1097
1098 if( evt->IsDrag( BUT_LEFT ) )
1099 {
1101 {
1102 if( m_selection.GetSize() > 0 )
1103 {
1104 anySubtracted = true;
1105 ClearSelection( true /*quiet mode*/ );
1106 }
1107 }
1108
1109 // Start drawing a selection box
1110 area.SetOrigin( evt->DragOrigin() );
1111 area.SetEnd( evt->Position() );
1114 area.SetExclusiveOr( false );
1115
1116 view->SetVisible( &area, true );
1117 view->Update( &area );
1118 getViewControls()->SetAutoPan( true );
1119 }
1120
1121 if( evt->IsMouseUp( BUT_LEFT ) )
1122 {
1123 getViewControls()->SetAutoPan( false );
1124
1125 // End drawing the selection box
1126 view->SetVisible( &area, false );
1127
1128 std::vector<KIGFX::VIEW::LAYER_ITEM_PAIR> candidates;
1129 BOX2I selectionBox = area.ViewBBox();
1130 view->Query( selectionBox, candidates ); // Get the list of nearby items
1131
1132 int height = area.GetEnd().y - area.GetOrigin().y;
1133
1134 // Construct a BOX2I to determine BOARD_ITEM selection
1135 BOX2I selectionRect( area.GetOrigin(), VECTOR2I( width, height ) );
1136
1137 selectionRect.Normalize();
1138
1139 GENERAL_COLLECTOR collector;
1140 GENERAL_COLLECTOR padsCollector;
1141 std::set<EDA_ITEM*> group_items;
1142
1143 for( PCB_GROUP* group : board()->Groups() )
1144 {
1145 // The currently entered group does not get limited
1146 if( m_enteredGroup == group )
1147 continue;
1148
1149 std::unordered_set<EDA_ITEM*>& newset = group->GetItems();
1150
1151 // If we are not greedy and have selected the whole group, add just one item
1152 // to allow it to be promoted to the group later
1153 if( !greedySelection && selectionRect.Contains( group->GetBoundingBox() )
1154 && newset.size() )
1155 {
1156 for( EDA_ITEM* group_item : newset )
1157 {
1158 if( Selectable( static_cast<BOARD_ITEM*>( group_item ) ) )
1159 collector.Append( *newset.begin() );
1160 }
1161 }
1162
1163 for( EDA_ITEM* group_item : newset )
1164 group_items.emplace( group_item );
1165 }
1166
1167 for( const KIGFX::VIEW::LAYER_ITEM_PAIR& candidate : candidates )
1168 {
1169 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( candidate.first );
1170
1171 if( item && Selectable( item ) && item->HitTest( selectionRect, !greedySelection )
1172 && ( greedySelection || !group_items.count( item ) ) )
1173 {
1174 if( item->Type() == PCB_PAD_T && !m_isFootprintEditor )
1175 padsCollector.Append( item );
1176 else
1177 collector.Append( item );
1178 }
1179 }
1180
1181 // Apply the stateful filter
1182 FilterCollectedItems( collector, true );
1183
1184 FilterCollectorForHierarchy( collector, true );
1185
1186 // If we selected nothing but pads, allow them to be selected
1187 if( collector.GetCount() == 0 )
1188 {
1189 collector = padsCollector;
1190 FilterCollectedItems( collector, true );
1191 FilterCollectorForHierarchy( collector, true );
1192 }
1193
1194 // Sort the filtered selection by rows and columns to have a nice default
1195 // for tools that can use it.
1196 std::sort( collector.begin(), collector.end(),
1197 []( EDA_ITEM* a, EDA_ITEM* b )
1198 {
1199 VECTOR2I aPos = a->GetPosition();
1200 VECTOR2I bPos = b->GetPosition();
1201
1202 if( aPos.y == bPos.y )
1203 return aPos.x < bPos.x;
1204
1205 return aPos.y < bPos.y;
1206 } );
1207
1208 for( EDA_ITEM* i : collector )
1209 {
1210 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( i );
1211
1212 if( m_subtractive || ( m_exclusive_or && item->IsSelected() ) )
1213 {
1214 unselect( item );
1215 anySubtracted = true;
1216 }
1217 else
1218 {
1219 select( item );
1220 anyAdded = true;
1221 }
1222 }
1223
1224 m_selection.SetIsHover( false );
1225
1226 // Inform other potentially interested tools
1227 if( anyAdded )
1229 else if( anySubtracted )
1231
1232 break; // Stop waiting for events
1233 }
1234
1235 // Allow some actions for navigation
1236 for( int i = 0; allowedActions[i]; ++i )
1237 {
1238 if( evt->IsAction( allowedActions[i] ) )
1239 {
1240 evt->SetPassEvent();
1241 break;
1242 }
1243 }
1244 }
1245
1246 getViewControls()->SetAutoPan( false );
1247
1248 // Stop drawing the selection box
1249 view->Remove( &area );
1250 m_multiple = false; // Multiple selection mode is inactive
1251
1252 if( !cancelled )
1254
1256
1257 return cancelled;
1258}
1259
1260
1262{
1263 wxMouseState keyboardState = wxGetMouseState();
1264
1265 setModifiersState( keyboardState.ShiftDown(), keyboardState.ControlDown(),
1266 keyboardState.AltDown() );
1267
1268 m_skip_heuristics = true;
1270 m_skip_heuristics = false;
1271
1272 return 0;
1273}
1274
1275
1276
1278{
1280
1281 selectCursor( false, aClientFilter );
1282
1283 return 0;
1284}
1285
1286
1288{
1290
1291 return 0;
1292}
1293
1294
1296{
1297 GENERAL_COLLECTOR collection;
1298 BOX2I selectionBox;
1299
1300 selectionBox.SetMaximum();
1301
1302 getView()->Query( selectionBox,
1303 [&]( KIGFX::VIEW_ITEM* viewItem ) -> bool
1304 {
1305 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( viewItem );
1306
1307 if( !item || !Selectable( item ) || !itemPassesFilter( item, true ) )
1308 return true;
1309
1310 collection.Append( item );
1311 return true;
1312 } );
1313
1314 FilterCollectorForHierarchy( collection, true );
1315
1316 for( EDA_ITEM* item : collection )
1317 select( item );
1318
1320
1322
1323 return 0;
1324}
1325
1326
1328{
1329 BOX2I selectionBox;
1330
1331 selectionBox.SetMaximum();
1332
1333 getView()->Query( selectionBox,
1334 [&]( KIGFX::VIEW_ITEM* viewItem ) -> bool
1335 {
1336 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( viewItem );
1337
1338 if( !item || !Selectable( item ) )
1339 return true;
1340
1341 unselect( item );
1342 return true;
1343 } );
1344
1346
1348
1349 return 0;
1350}
1351
1352
1354 PCB_SELECTION_TOOL* sTool )
1355{
1356 // Narrow the collection down to a single BOARD_CONNECTED_ITEM for each represented net.
1357 // All other items types are removed.
1358 std::set<int> representedNets;
1359
1360 for( int i = aCollector.GetCount() - 1; i >= 0; i-- )
1361 {
1362 BOARD_CONNECTED_ITEM* item = dynamic_cast<BOARD_CONNECTED_ITEM*>( aCollector[i] );
1363
1364 if( !item )
1365 aCollector.Remove( i );
1366 else if ( representedNets.count( item->GetNetCode() ) )
1367 aCollector.Remove( i );
1368 else
1369 representedNets.insert( item->GetNetCode() );
1370 }
1371}
1372
1373
1375{
1376 std::deque<EDA_ITEM*> selectedItems = m_selection.GetItems();
1377
1378 // Get all footprints and pads
1379 std::vector<BOARD_CONNECTED_ITEM*> toUnroute;
1380
1381 for( EDA_ITEM* item : selectedItems )
1382 {
1383 if( item->Type() == PCB_FOOTPRINT_T )
1384 {
1385 for( PAD* pad : static_cast<FOOTPRINT*>( item )->Pads() )
1386 toUnroute.push_back( pad );
1387 }
1388 else if( BOARD_CONNECTED_ITEM::ClassOf( item ) )
1389 {
1390 toUnroute.push_back( static_cast<BOARD_CONNECTED_ITEM*>( item ) );
1391 }
1392 }
1393
1394 // Clear selection so we don't delete our footprints/pads
1395 ClearSelection( true );
1396
1397 // Get the tracks on our list of pads, then delete them
1400
1401 // Reselect our footprint/pads as they were in our original selection
1402 for( EDA_ITEM* item : selectedItems )
1403 {
1404 if( item->Type() == PCB_FOOTPRINT_T || item->Type() == PCB_PAD_T )
1405 select( item );
1406 }
1407
1408 return 0;
1409}
1410
1411
1413{
1414 // expandConnection will get called no matter whether the user selected a connected item or a
1415 // non-connected shape (graphic on a non-copper layer). The algorithm for expanding to connected
1416 // items is different from graphics, so they need to be handled separately.
1417 unsigned initialCount = 0;
1418
1419 for( const EDA_ITEM* item : m_selection.GetItems() )
1420 {
1421 if( item->Type() == PCB_FOOTPRINT_T
1422 || item->Type() == PCB_GENERATOR_T
1423 || ( static_cast<const BOARD_ITEM*>( item )->IsConnected() ) )
1424 {
1425 initialCount++;
1426 }
1427 }
1428
1429 if( initialCount == 0 )
1430 {
1431 // First, process any graphic shapes we have
1432 std::vector<PCB_SHAPE*> startShapes;
1433
1434 for( EDA_ITEM* item : m_selection.GetItems() )
1435 {
1436 if( isExpandableGraphicShape( item ) )
1437 startShapes.push_back( static_cast<PCB_SHAPE*>( item ) );
1438 }
1439
1440 // If no non-copper shapes; fall back to looking for connected items
1441 if( !startShapes.empty() )
1442 selectAllConnectedShapes( startShapes );
1443 else
1445 }
1446
1447 m_frame->SetStatusText( _( "Select/Expand Connection..." ) );
1448
1449 for( STOP_CONDITION stopCondition : { STOP_AT_JUNCTION, STOP_AT_PAD, STOP_NEVER } )
1450 {
1451 std::deque<EDA_ITEM*> selectedItems = m_selection.GetItems();
1452
1453 for( EDA_ITEM* item : selectedItems )
1454 item->ClearTempFlags();
1455
1456 std::vector<BOARD_CONNECTED_ITEM*> startItems;
1457
1458 for( EDA_ITEM* item : selectedItems )
1459 {
1460 if( item->Type() == PCB_FOOTPRINT_T )
1461 {
1462 FOOTPRINT* footprint = static_cast<FOOTPRINT*>( item );
1463
1464 for( PAD* pad : footprint->Pads() )
1465 startItems.push_back( pad );
1466 }
1467 else if( item->Type() == PCB_GENERATOR_T )
1468 {
1469 for( BOARD_ITEM* generatedItem : static_cast<PCB_GENERATOR*>( item )->GetBoardItems() )
1470 {
1471 if( BOARD_CONNECTED_ITEM::ClassOf( generatedItem ) )
1472 startItems.push_back( static_cast<BOARD_CONNECTED_ITEM*>( generatedItem ) );
1473 }
1474 }
1475 else if( BOARD_CONNECTED_ITEM::ClassOf( item ) )
1476 {
1477 startItems.push_back( static_cast<BOARD_CONNECTED_ITEM*>( item ) );
1478 }
1479 }
1480
1481 selectAllConnectedTracks( startItems, stopCondition );
1482
1483 if( m_selection.GetItems().size() > initialCount )
1484 break;
1485 }
1486
1487 m_frame->SetStatusText( wxEmptyString );
1488
1489 // Inform other potentially interested tools
1491
1492 return 0;
1493}
1494
1495
1497 const std::vector<BOARD_CONNECTED_ITEM*>& aStartItems, STOP_CONDITION aStopCondition )
1498{
1499 const LSET allCuMask = LSET::AllCuMask();
1500
1501 PROF_TIMER refreshTimer;
1502 double refreshIntervalMs = 500; // Refresh display with this interval to indicate progress
1503 int lastSelectionSize = (int) m_selection.GetSize();
1504
1505 auto connectivity = board()->GetConnectivity();
1506
1507 std::map<VECTOR2I, std::vector<PCB_TRACK*>> trackMap;
1508 std::map<VECTOR2I, PCB_VIA*> viaMap;
1509 std::map<VECTOR2I, PAD*> padMap;
1510 std::map<VECTOR2I, std::vector<PCB_SHAPE*>> shapeMap;
1511 std::set<PAD*> startPadSet;
1512 std::vector<BOARD_CONNECTED_ITEM*> cleanupItems;
1513 std::vector<std::pair<VECTOR2I, LSET>> activePts;
1514
1515 for( BOARD_CONNECTED_ITEM* startItem : aStartItems )
1516 {
1517 // Register starting pads
1518 if( startItem->Type() == PCB_PAD_T )
1519 startPadSet.insert( static_cast<PAD*>( startItem ) );
1520
1521 // Select any starting track items
1522 if( startItem->IsType( { PCB_TRACE_T, PCB_ARC_T, PCB_VIA_T } ) )
1523 select( startItem );
1524 }
1525
1526 for( BOARD_CONNECTED_ITEM* startItem : aStartItems )
1527 {
1528 if( startItem->HasFlag( SKIP_STRUCT ) ) // Skip already visited items
1529 continue;
1530
1531 auto connectedItems = connectivity->GetConnectedItems( startItem,
1533
1534 // Build maps of connected items
1535 for( BOARD_CONNECTED_ITEM* item : connectedItems )
1536 {
1537 switch( item->Type() )
1538 {
1539 case PCB_ARC_T:
1540 case PCB_TRACE_T:
1541 {
1542 PCB_TRACK* track = static_cast<PCB_TRACK*>( item );
1543 trackMap[track->GetStart()].push_back( track );
1544 trackMap[track->GetEnd()].push_back( track );
1545 break;
1546 }
1547
1548 case PCB_VIA_T:
1549 {
1550 PCB_VIA* via = static_cast<PCB_VIA*>( item );
1551 viaMap[via->GetStart()] = via;
1552 break;
1553 }
1554
1555 case PCB_PAD_T:
1556 {
1557 PAD* pad = static_cast<PAD*>( item );
1558 padMap[pad->GetPosition()] = pad;
1559 break;
1560 }
1561
1562 case PCB_SHAPE_T:
1563 {
1564 PCB_SHAPE* shape = static_cast<PCB_SHAPE*>( item );
1565
1566 for( const auto& point : shape->GetConnectionPoints() )
1567 shapeMap[point].push_back( shape );
1568
1569 break;
1570 }
1571
1572 default:
1573 break;
1574 }
1575 }
1576
1577 // Set up the initial active points
1578 switch( startItem->Type() )
1579 {
1580 case PCB_ARC_T:
1581 case PCB_TRACE_T:
1582 {
1583 PCB_TRACK* track = static_cast<PCB_TRACK*>( startItem );
1584
1585 activePts.push_back( { track->GetStart(), track->GetLayerSet() } );
1586 activePts.push_back( { track->GetEnd(), track->GetLayerSet() } );
1587 break;
1588 }
1589
1590 case PCB_VIA_T:
1591 activePts.push_back( { startItem->GetPosition(), startItem->GetLayerSet() } );
1592 break;
1593
1594 case PCB_PAD_T:
1595 activePts.push_back( { startItem->GetPosition(), startItem->GetLayerSet() } );
1596 break;
1597
1598 case PCB_SHAPE_T:
1599 {
1600 PCB_SHAPE* shape = static_cast<PCB_SHAPE*>( startItem );
1601
1602 for( const auto& point : shape->GetConnectionPoints() )
1603 activePts.push_back( { point, startItem->GetLayerSet() } );
1604
1605 break;
1606 }
1607
1608 default:
1609 break;
1610 }
1611
1612 bool expand = true;
1613 int failSafe = 0;
1614
1615 // Iterative push from all active points
1616 while( expand && failSafe++ < 100000 )
1617 {
1618 expand = false;
1619
1620 for( int i = (int) activePts.size() - 1; i >= 0; --i )
1621 {
1622 VECTOR2I pt = activePts[i].first;
1623 LSET layerSetCu = activePts[i].second & allCuMask;
1624
1625 auto viaIt = viaMap.find( pt );
1626 auto padIt = padMap.find( pt );
1627
1628 bool gotVia = ( viaIt != viaMap.end() )
1629 && ( layerSetCu & ( viaIt->second->GetLayerSet() ) ).any();
1630
1631 bool gotPad = ( padIt != padMap.end() )
1632 && ( layerSetCu & ( padIt->second->GetLayerSet() ) ).any();
1633
1634 bool gotNonStartPad =
1635 gotPad && ( startPadSet.find( padIt->second ) == startPadSet.end() );
1636
1637 if( aStopCondition == STOP_AT_JUNCTION )
1638 {
1639 size_t pt_count = 0;
1640
1641 for( PCB_TRACK* track : trackMap[pt] )
1642 {
1643 if( track->GetStart() != track->GetEnd()
1644 && layerSetCu.Contains( track->GetLayer() ) )
1645 {
1646 pt_count++;
1647 }
1648 }
1649
1650 if( pt_count > 2 || gotVia || gotNonStartPad )
1651 {
1652 activePts.erase( activePts.begin() + i );
1653 continue;
1654 }
1655 }
1656 else if( aStopCondition == STOP_AT_PAD )
1657 {
1658 if( gotNonStartPad )
1659 {
1660 activePts.erase( activePts.begin() + i );
1661 continue;
1662 }
1663 }
1664
1665 if( gotPad )
1666 {
1667 PAD* pad = padIt->second;
1668
1669 if( !pad->HasFlag( SKIP_STRUCT ) )
1670 {
1671 pad->SetFlags( SKIP_STRUCT );
1672 cleanupItems.push_back( pad );
1673
1674 activePts.push_back( { pad->GetPosition(), pad->GetLayerSet() } );
1675 expand = true;
1676 }
1677 }
1678
1679 for( PCB_TRACK* track : trackMap[pt] )
1680 {
1681 if( !layerSetCu.Contains( track->GetLayer() ) )
1682 continue;
1683
1684 if( !track->IsSelected() )
1685 select( track );
1686
1687 if( !track->HasFlag( SKIP_STRUCT ) )
1688 {
1689 track->SetFlags( SKIP_STRUCT );
1690 cleanupItems.push_back( track );
1691
1692 if( track->GetStart() == pt )
1693 activePts.push_back( { track->GetEnd(), track->GetLayerSet() } );
1694 else
1695 activePts.push_back( { track->GetStart(), track->GetLayerSet() } );
1696
1697 expand = true;
1698 }
1699 }
1700
1701 for( PCB_SHAPE* shape : shapeMap[pt] )
1702 {
1703 if( !layerSetCu.Contains( shape->GetLayer() ) )
1704 continue;
1705
1706 if( !shape->IsSelected() )
1707 select( shape );
1708
1709 if( !shape->HasFlag( SKIP_STRUCT ) )
1710 {
1711 shape->SetFlags( SKIP_STRUCT );
1712 cleanupItems.push_back( shape );
1713
1714 for( const VECTOR2I& newPoint : shape->GetConnectionPoints() )
1715 {
1716 if( newPoint == pt )
1717 continue;
1718
1719 activePts.push_back( { newPoint, shape->GetLayerSet() } );
1720 }
1721
1722 expand = true;
1723 }
1724 }
1725
1726 if( viaMap.count( pt ) )
1727 {
1728 PCB_VIA* via = viaMap[pt];
1729
1730 if( !via->IsSelected() )
1731 select( via );
1732
1733 if( !via->HasFlag( SKIP_STRUCT ) )
1734 {
1735 via->SetFlags( SKIP_STRUCT );
1736 cleanupItems.push_back( via );
1737
1738 activePts.push_back( { via->GetPosition(), via->GetLayerSet() } );
1739 expand = true;
1740 }
1741 }
1742
1743 activePts.erase( activePts.begin() + i );
1744 }
1745
1746 // Refresh display for the feel of progress
1747 if( refreshTimer.msecs() >= refreshIntervalMs )
1748 {
1749 if( m_selection.Size() != lastSelectionSize )
1750 {
1752 lastSelectionSize = m_selection.Size();
1753 }
1754
1755 refreshTimer.Start();
1756 }
1757 }
1758 }
1759
1760 std::set<EDA_ITEM*> toDeselect;
1761 std::set<EDA_ITEM*> toSelect;
1762
1763 // Promote generated members to their PCB_GENERATOR parents
1764 for( EDA_ITEM* item : m_selection )
1765 {
1766 if( !item->IsBOARD_ITEM() )
1767 continue;
1768
1769 BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( item );
1770 EDA_GROUP* parent = boardItem->GetParentGroup();
1771
1772 if( parent && parent->AsEdaItem()->Type() == PCB_GENERATOR_T )
1773 {
1774 toDeselect.insert( item );
1775
1776 if( !parent->AsEdaItem()->IsSelected() )
1777 toSelect.insert( parent->AsEdaItem() );
1778 }
1779 }
1780
1781 for( EDA_ITEM* item : toDeselect )
1782 unselect( item );
1783
1784 for( EDA_ITEM* item : toSelect )
1785 select( item );
1786
1787 for( BOARD_CONNECTED_ITEM* item : cleanupItems )
1788 item->ClearFlags( SKIP_STRUCT );
1789}
1790
1791
1793{
1794 if( aItem->Type() == PCB_SHAPE_T )
1795 {
1796 const PCB_SHAPE* shape = static_cast<const PCB_SHAPE*>( aItem );
1797
1798 switch( shape->GetShape() )
1799 {
1800 case SHAPE_T::SEGMENT:
1801 case SHAPE_T::ARC:
1802 case SHAPE_T::BEZIER:
1803 return !shape->IsOnCopperLayer();
1804
1805 case SHAPE_T::POLY:
1806 return !shape->IsOnCopperLayer() && !shape->IsClosed();
1807
1808 default:
1809 return false;
1810 }
1811 }
1812
1813 return false;
1814}
1815
1816
1817void PCB_SELECTION_TOOL::selectAllConnectedShapes( const std::vector<PCB_SHAPE*>& aStartItems )
1818{
1819 std::stack<PCB_SHAPE*> toSearch;
1820 std::set<PCB_SHAPE*> toCleanup;
1821
1822 for( PCB_SHAPE* startItem : aStartItems )
1823 toSearch.push( startItem );
1824
1825 GENERAL_COLLECTOR collector;
1827
1828 auto searchPoint = [&]( const VECTOR2I& aWhere )
1829 {
1830 collector.Collect( board(), { PCB_SHAPE_T }, aWhere, guide );
1831
1832 for( EDA_ITEM* item : collector )
1833 {
1834 if( isExpandableGraphicShape( item ) )
1835 toSearch.push( static_cast<PCB_SHAPE*>( item ) );
1836 }
1837 };
1838
1839 while( !toSearch.empty() )
1840 {
1841 PCB_SHAPE* shape = toSearch.top();
1842 toSearch.pop();
1843
1844 if( shape->HasFlag( SKIP_STRUCT ) )
1845 continue;
1846
1847 select( shape );
1848 shape->SetFlags( SKIP_STRUCT );
1849 toCleanup.insert( shape );
1850
1851 guide.SetLayerVisibleBits( shape->GetLayerSet() );
1852
1853 searchPoint( shape->GetStart() );
1854 searchPoint( shape->GetEnd() );
1855 }
1856
1857 for( PCB_SHAPE* shape : toCleanup )
1858 shape->ClearFlags( SKIP_STRUCT );
1859}
1860
1861
1863{
1864 // Get all pads
1865 std::vector<PAD*> pads;
1866
1867 for( EDA_ITEM* item : m_selection.GetItems() )
1868 {
1869 if( item->Type() == PCB_FOOTPRINT_T )
1870 {
1871 for( PAD* pad : static_cast<FOOTPRINT*>( item )->Pads() )
1872 pads.push_back( pad );
1873 }
1874 else if( item->Type() == PCB_PAD_T )
1875 {
1876 pads.push_back( static_cast<PAD*>( item ) );
1877 }
1878 }
1879
1880 // Select every footprint on the end of the ratsnest for each pad in our selection
1881 std::shared_ptr<CONNECTIVITY_DATA> conn = board()->GetConnectivity();
1882
1883 for( PAD* pad : pads )
1884 {
1885 for( const CN_EDGE& edge : conn->GetRatsnestForPad( pad ) )
1886 {
1887 wxCHECK2( edge.GetSourceNode() && !edge.GetSourceNode()->Dirty(), continue );
1888 wxCHECK2( edge.GetTargetNode() && !edge.GetTargetNode()->Dirty(), continue );
1889
1890 BOARD_CONNECTED_ITEM* sourceParent = edge.GetSourceNode()->Parent();
1891 BOARD_CONNECTED_ITEM* targetParent = edge.GetTargetNode()->Parent();
1892
1893 if( sourceParent == pad )
1894 {
1895 if( targetParent->Type() == PCB_PAD_T )
1896 select( static_cast<PAD*>( targetParent )->GetParent() );
1897 }
1898 else if( targetParent == pad )
1899 {
1900 if( sourceParent->Type() == PCB_PAD_T )
1901 select( static_cast<PAD*>( sourceParent )->GetParent() );
1902 }
1903 }
1904 }
1905
1906 return 0;
1907}
1908
1909
1911{
1912 PCB_SELECTION originalSelection = m_selection;
1913
1914 // Get all pads
1915 std::vector<PAD*> pads;
1916
1917 for( EDA_ITEM* item : m_selection.GetItems() )
1918 {
1919 if( item->Type() == PCB_FOOTPRINT_T )
1920 {
1921 for( PAD* pad : static_cast<FOOTPRINT*>( item )->Pads() )
1922 pads.push_back( pad );
1923 }
1924 else if( item->Type() == PCB_PAD_T )
1925 {
1926 pads.push_back( static_cast<PAD*>( item ) );
1927 }
1928 }
1929
1931
1932 // Select every footprint on the end of the ratsnest for each pad in our selection
1933 std::shared_ptr<CONNECTIVITY_DATA> conn = board()->GetConnectivity();
1934
1935 for( PAD* pad : pads )
1936 {
1937 const std::vector<CN_EDGE> edges = conn->GetRatsnestForPad( pad );
1938
1939 // Need to have something unconnected to grab
1940 if( edges.size() == 0 )
1941 continue;
1942
1943 double currentDistance = DBL_MAX;
1944 FOOTPRINT* nearest = nullptr;
1945
1946 // Check every ratsnest line for the nearest one
1947 for( const CN_EDGE& edge : edges )
1948 {
1949 if( edge.GetSourceNode()->Parent()->GetParentFootprint()
1950 == edge.GetTargetNode()->Parent()->GetParentFootprint() )
1951 {
1952 continue; // This edge is a loop on the same footprint
1953 }
1954
1955 // Figure out if we are the source or the target node on the ratnest
1956 const CN_ANCHOR* other = edge.GetSourceNode()->Parent() == pad ? edge.GetTargetNode().get()
1957 : edge.GetSourceNode().get();
1958
1959 wxCHECK2( other && !other->Dirty(), continue );
1960
1961 // We only want to grab footprints, so the ratnest has to point to a pad
1962 if( other->Parent()->Type() != PCB_PAD_T )
1963 continue;
1964
1965 if( edge.GetLength() < currentDistance )
1966 {
1967 currentDistance = edge.GetLength();
1968 nearest = other->Parent()->GetParentFootprint();
1969 }
1970 }
1971
1972 if( nearest != nullptr )
1973 select( nearest );
1974 }
1975
1977
1978 return 0;
1979}
1980
1981
1982void PCB_SELECTION_TOOL::SelectAllItemsOnNet( int aNetCode, bool aSelect )
1983{
1984 std::shared_ptr<CONNECTIVITY_DATA> conn = board()->GetConnectivity();
1985
1986 for( BOARD_ITEM* item : conn->GetNetItems( aNetCode, { PCB_TRACE_T,
1987 PCB_ARC_T,
1988 PCB_VIA_T,
1989 PCB_SHAPE_T } ) )
1990 {
1991 if( itemPassesFilter( item, true ) )
1992 aSelect ? select( item ) : unselect( item );
1993 }
1994}
1995
1996
1998{
1999 bool select = aEvent.IsAction( &PCB_ACTIONS::selectNet );
2000
2001 // If we've been passed an argument, just select that netcode1
2002 int netcode = aEvent.Parameter<int>();
2003
2004 if( netcode > 0 )
2005 {
2006 SelectAllItemsOnNet( netcode, select );
2007
2008 // Inform other potentially interested tools
2009 if( m_selection.Size() > 0 )
2011 else
2013
2014 return 0;
2015 }
2016
2017 if( !selectCursor() )
2018 return 0;
2019
2020 // copy the selection, since we're going to iterate and modify
2022
2023 for( EDA_ITEM* i : selection )
2024 {
2025 BOARD_CONNECTED_ITEM* connItem = dynamic_cast<BOARD_CONNECTED_ITEM*>( i );
2026
2027 if( connItem )
2028 SelectAllItemsOnNet( connItem->GetNetCode(), select );
2029 }
2030
2031 // Inform other potentially interested tools
2032 if( m_selection.Size() > 0 )
2034 else
2036
2037 return 0;
2038}
2039
2040
2042{
2043 std::vector<BOARD_ITEM*> footprints;
2044
2045 // store all footprints that are on that sheet path
2046 for( FOOTPRINT* footprint : board()->Footprints() )
2047 {
2048 if( footprint == nullptr )
2049 continue;
2050
2051 wxString footprint_path = footprint->GetPath().AsString().BeforeLast( '/' );
2052
2053 if( footprint_path.IsEmpty() )
2054 footprint_path += '/';
2055
2056 if( footprint_path == aSheetPath )
2057 footprints.push_back( footprint );
2058 }
2059
2060 for( BOARD_ITEM* i : footprints )
2061 {
2062 if( i != nullptr )
2063 select( i );
2064 }
2065
2066 selectConnections( footprints );
2067}
2068
2069
2070void PCB_SELECTION_TOOL::selectConnections( const std::vector<BOARD_ITEM*>& aItems )
2071{
2072 // Generate a list of all pads, and of all nets they belong to.
2073 std::list<int> netcodeList;
2074 std::vector<BOARD_CONNECTED_ITEM*> padList;
2075
2076 for( BOARD_ITEM* item : aItems )
2077 {
2078 switch( item->Type() )
2079 {
2080 case PCB_FOOTPRINT_T:
2081 {
2082 for( PAD* pad : static_cast<FOOTPRINT*>( item )->Pads() )
2083 {
2084 if( pad->IsConnected() )
2085 {
2086 netcodeList.push_back( pad->GetNetCode() );
2087 padList.push_back( pad );
2088 }
2089 }
2090
2091 break;
2092 }
2093
2094 case PCB_PAD_T:
2095 {
2096 PAD* pad = static_cast<PAD*>( item );
2097
2098 if( pad->IsConnected() )
2099 {
2100 netcodeList.push_back( pad->GetNetCode() );
2101 padList.push_back( pad );
2102 }
2103
2104 break;
2105 }
2106
2107 default:
2108 break;
2109 }
2110 }
2111
2112 // Sort for binary search
2113 std::sort( padList.begin(), padList.end() );
2114
2115 // remove all duplicates
2116 netcodeList.sort();
2117 netcodeList.unique();
2118
2120
2121 // now we need to find all footprints that are connected to each of these nets then we need
2122 // to determine if these footprints are in the list of footprints
2123 std::vector<int> removeCodeList;
2124 std::shared_ptr<CONNECTIVITY_DATA> conn = board()->GetConnectivity();
2125
2126 for( int netCode : netcodeList )
2127 {
2128 for( BOARD_CONNECTED_ITEM* pad : conn->GetNetItems( netCode, { PCB_PAD_T } ) )
2129 {
2130 if( !std::binary_search( padList.begin(), padList.end(), pad ) )
2131 {
2132 // if we cannot find the pad in the padList then we can assume that that pad
2133 // should not be used, therefore invalidate this netcode.
2134 removeCodeList.push_back( netCode );
2135 break;
2136 }
2137 }
2138 }
2139
2140 for( int removeCode : removeCodeList )
2141 netcodeList.remove( removeCode );
2142
2143 std::unordered_set<BOARD_ITEM*> localConnectionList;
2144
2145 for( int netCode : netcodeList )
2146 {
2147 for( BOARD_ITEM* item : conn->GetNetItems( netCode, { PCB_TRACE_T,
2148 PCB_ARC_T,
2149 PCB_VIA_T,
2150 PCB_SHAPE_T } ) )
2151 {
2152 localConnectionList.insert( item );
2153 }
2154 }
2155
2156 for( BOARD_ITEM* item : localConnectionList )
2157 select( item );
2158}
2159
2160
2162{
2163 std::vector<BOARD_ITEM*>* items = aEvent.Parameter<std::vector<BOARD_ITEM*>*>();
2164
2165 if( items )
2166 doSyncSelection( *items, false );
2167
2168 return 0;
2169}
2170
2171
2173{
2174 std::vector<BOARD_ITEM*>* items = aEvent.Parameter<std::vector<BOARD_ITEM*>*>();
2175
2176 if( items )
2177 doSyncSelection( *items, true );
2178
2179 return 0;
2180}
2181
2182
2183void PCB_SELECTION_TOOL::doSyncSelection( const std::vector<BOARD_ITEM*>& aItems, bool aWithNets )
2184{
2186 return;
2187
2188 ClearSelection( true /*quiet mode*/ );
2189
2190 // Perform individual selection of each item before processing the event.
2191 for( BOARD_ITEM* item : aItems )
2192 select( item );
2193
2194 if( aWithNets )
2195 selectConnections( aItems );
2196
2198
2199 if( bbox.GetWidth() != 0 && bbox.GetHeight() != 0 )
2200 {
2202 {
2204 ZoomFitCrossProbeBBox( bbox );
2205
2206 m_frame->FocusOnLocation( bbox.Centre() );
2207 }
2208 }
2209
2211
2213
2214 if( m_selection.Size() > 0 )
2216}
2217
2218
2220{
2221 ClearSelection( true /*quiet mode*/ );
2222 wxString sheetPath = *aEvent.Parameter<wxString*>();
2223
2224 selectAllItemsOnSheet( sheetPath );
2225
2227
2228 if( m_selection.Size() > 0 )
2230
2231 return 0;
2232}
2233
2234
2236{
2237 // this function currently only supports footprints since they are only on one sheet.
2238 EDA_ITEM* item = m_selection.Front();
2239
2240 if( !item )
2241 return 0;
2242
2243 if( item->Type() != PCB_FOOTPRINT_T )
2244 return 0;
2245
2246 FOOTPRINT* footprint = dynamic_cast<FOOTPRINT*>( item );
2247
2248 if( !footprint || footprint->GetPath().empty() )
2249 return 0;
2250
2251 ClearSelection( true /*quiet mode*/ );
2252
2253 // get the sheet path only.
2254 wxString sheetPath = footprint->GetPath().AsString().BeforeLast( '/' );
2255
2256 if( sheetPath.IsEmpty() )
2257 sheetPath += '/';
2258
2259 selectAllItemsOnSheet( sheetPath );
2260
2261 // Inform other potentially interested tools
2262 if( m_selection.Size() > 0 )
2264
2265 return 0;
2266}
2267
2268
2270{
2271 // Should recalculate the view to zoom in on the selection.
2272 BOX2I selectionBox = m_selection.GetBoundingBox();
2274
2275 VECTOR2D screenSize = view->ToWorld( ToVECTOR2D( m_frame->GetCanvas()->GetClientSize() ),
2276 false );
2277 screenSize.x = std::max( 10.0, screenSize.x );
2278 screenSize.y = std::max( 10.0, screenSize.y );
2279
2280 if( selectionBox.GetWidth() != 0 || selectionBox.GetHeight() != 0 )
2281 {
2282 VECTOR2D vsize = selectionBox.GetSize();
2283 double scale = view->GetScale() / std::max( fabs( vsize.x / screenSize.x ),
2284 fabs( vsize.y / screenSize.y ) );
2285 view->SetScale( scale );
2286 view->SetCenter( selectionBox.Centre() );
2287 view->Add( &m_selection );
2288 }
2289
2291}
2292
2293
2295{
2296 // Should recalculate the view to zoom in on the bbox.
2298
2299 if( aBBox.GetWidth() == 0 )
2300 return;
2301
2302 BOX2I bbox = aBBox;
2303 bbox.Normalize();
2304
2305 //#define DEFAULT_PCBNEW_CODE // Un-comment for normal full zoom KiCad algorithm
2306#ifdef DEFAULT_PCBNEW_CODE
2307 auto bbSize = bbox.Inflate( bbox.GetWidth() * 0.2f ).GetSize();
2308 auto screenSize = view->ToWorld( GetCanvas()->GetClientSize(), false );
2309
2310 // The "fabs" on x ensures the right answer when the view is flipped
2311 screenSize.x = std::max( 10.0, fabs( screenSize.x ) );
2312 screenSize.y = std::max( 10.0, screenSize.y );
2313 double ratio = std::max( fabs( bbSize.x / screenSize.x ), fabs( bbSize.y / screenSize.y ) );
2314
2315 // Try not to zoom on every cross-probe; it gets very noisy
2316 if( crossProbingSettings.zoom_to_fit && ( ratio < 0.5 || ratio > 1.0 ) )
2317 view->SetScale( view->GetScale() / ratio );
2318#endif // DEFAULT_PCBNEW_CODE
2319
2320#ifndef DEFAULT_PCBNEW_CODE // Do the scaled zoom
2321 auto bbSize = bbox.Inflate( KiROUND( bbox.GetWidth() * 0.2 ) ).GetSize();
2322 VECTOR2D screenSize = view->ToWorld( ToVECTOR2D( m_frame->GetCanvas()->GetClientSize() ),
2323 false );
2324
2325 // This code tries to come up with a zoom factor that doesn't simply zoom in
2326 // to the cross probed component, but instead shows a reasonable amount of the
2327 // circuit around it to provide context. This reduces or eliminates the need
2328 // to manually change the zoom because it's too close.
2329
2330 // Using the default text height as a constant to compare against, use the
2331 // height of the bounding box of visible items for a footprint to figure out
2332 // if this is a big footprint (like a processor) or a small footprint (like a resistor).
2333 // This ratio is not useful by itself as a scaling factor. It must be "bent" to
2334 // provide good scaling at varying component sizes. Bigger components need less
2335 // scaling than small ones.
2336 double currTextHeight = pcbIUScale.mmToIU( DEFAULT_TEXT_SIZE );
2337
2338 double compRatio = bbSize.y / currTextHeight; // Ratio of component to text height
2339
2340 // This will end up as the scaling factor we apply to "ratio".
2341 double compRatioBent = 1.0;
2342
2343 // This is similar to the original KiCad code that scaled the zoom to make sure
2344 // components were visible on screen. It's simply a ratio of screen size to
2345 // component size, and its job is to zoom in to make the component fullscreen.
2346 // Earlier in the code the component BBox is given a 20% margin to add some
2347 // breathing room. We compare the height of this enlarged component bbox to the
2348 // default text height. If a component will end up with the sides clipped, we
2349 // adjust later to make sure it fits on screen.
2350 //
2351 // The "fabs" on x ensures the right answer when the view is flipped
2352 screenSize.x = std::max( 10.0, fabs( screenSize.x ) );
2353 screenSize.y = std::max( 10.0, screenSize.y );
2354 double ratio = std::max( -1.0, fabs( bbSize.y / screenSize.y ) );
2355
2356 // Original KiCad code for how much to scale the zoom
2357 double kicadRatio = std::max( fabs( bbSize.x / screenSize.x ),
2358 fabs( bbSize.y / screenSize.y ) );
2359
2360 // LUT to scale zoom ratio to provide reasonable schematic context. Must work
2361 // with footprints of varying sizes (e.g. 0402 package and 200 pin BGA).
2362 // "first" is used as the input and "second" as the output
2363 //
2364 // "first" = compRatio (footprint height / default text height)
2365 // "second" = Amount to scale ratio by
2366 std::vector<std::pair<double, double>> lut {
2367 { 1, 8 },
2368 { 1.5, 5 },
2369 { 3, 3 },
2370 { 4.5, 2.5 },
2371 { 8, 2.0 },
2372 { 12, 1.7 },
2373 { 16, 1.5 },
2374 { 24, 1.3 },
2375 { 32, 1.0 },
2376 };
2377
2378
2379 std::vector<std::pair<double, double>>::iterator it;
2380
2381 compRatioBent = lut.back().second; // Large component default
2382
2383 if( compRatio >= lut.front().first )
2384 {
2385 // Use LUT to do linear interpolation of "compRatio" within "first", then
2386 // use that result to linearly interpolate "second" which gives the scaling
2387 // factor needed.
2388
2389 for( it = lut.begin(); it < lut.end() - 1; it++ )
2390 {
2391 if( it->first <= compRatio && next( it )->first >= compRatio )
2392 {
2393 double diffx = compRatio - it->first;
2394 double diffn = next( it )->first - it->first;
2395
2396 compRatioBent = it->second + ( next( it )->second - it->second ) * diffx / diffn;
2397 break; // We have our interpolated value
2398 }
2399 }
2400 }
2401 else
2402 {
2403 compRatioBent = lut.front().second; // Small component default
2404 }
2405
2406 // If the width of the part we're probing is bigger than what the screen width will be
2407 // after the zoom, then punt and use the KiCad zoom algorithm since it guarantees the
2408 // part's width will be encompassed within the screen. This will apply to parts that
2409 // are much wider than they are tall.
2410
2411 if( bbSize.x > screenSize.x * ratio * compRatioBent )
2412 {
2413 // Use standard KiCad zoom algorithm for parts too wide to fit screen/
2414 ratio = kicadRatio;
2415 compRatioBent = 1.0; // Reset so we don't modify the "KiCad" ratio
2416 wxLogTrace( "CROSS_PROBE_SCALE",
2417 "Part TOO WIDE for screen. Using normal KiCad zoom ratio: %1.5f", ratio );
2418 }
2419
2420 // Now that "compRatioBent" holds our final scaling factor we apply it to the original
2421 // fullscreen zoom ratio to arrive at the final ratio itself.
2422 ratio *= compRatioBent;
2423
2424 bool alwaysZoom = false; // DEBUG - allows us to minimize zooming or not
2425
2426 // Try not to zoom on every cross-probe; it gets very noisy
2427 if( ( ratio < 0.5 || ratio > 1.0 ) || alwaysZoom )
2428 view->SetScale( view->GetScale() / ratio );
2429#endif // ifndef DEFAULT_PCBNEW_CODE
2430}
2431
2432
2434{
2435 bool cleared = false;
2436
2437 if( m_selection.GetSize() > 0 )
2438 {
2439 // Don't fire an event now; most of the time it will be redundant as we're about to
2440 // fire a SelectedEvent.
2441 cleared = true;
2442 ClearSelection( true /*quiet mode*/ );
2443 }
2444
2445 if( aItem )
2446 {
2447 switch( aItem->Type() )
2448 {
2449 case PCB_NETINFO_T:
2450 {
2451 int netCode = static_cast<NETINFO_ITEM*>( aItem )->GetNetCode();
2452
2453 if( netCode > 0 )
2454 {
2455 SelectAllItemsOnNet( netCode, true );
2456 m_frame->FocusOnLocation( aItem->GetCenter() );
2457 }
2458 break;
2459 }
2460
2461 default:
2462 select( aItem );
2463 m_frame->FocusOnLocation( aItem->GetPosition() );
2464 }
2465
2466 // If the item has a bounding box, then zoom out if needed
2467 if( aItem->GetBoundingBox().GetHeight() > 0 && aItem->GetBoundingBox().GetWidth() > 0 )
2468 {
2469 // This adds some margin
2470 double marginFactor = 2;
2471
2472 KIGFX::PCB_VIEW* pcbView = canvas()->GetView();
2473 BOX2D screenBox = pcbView->GetViewport();
2474 VECTOR2D screenSize = screenBox.GetSize();
2475 BOX2I screenRect = BOX2ISafe( screenBox.GetOrigin(), screenSize / marginFactor );
2476
2477 if( !screenRect.Contains( aItem->GetBoundingBox() ) )
2478 {
2479 double scaleX = screenSize.x /
2480 static_cast<double>( aItem->GetBoundingBox().GetWidth() );
2481 double scaleY = screenSize.y /
2482 static_cast<double>( aItem->GetBoundingBox().GetHeight() );
2483
2484 scaleX /= marginFactor;
2485 scaleY /= marginFactor;
2486
2487 double scale = scaleX > scaleY ? scaleY : scaleX;
2488
2489 if( scale < 1 ) // Don't zoom in, only zoom out
2490 {
2491 pcbView->SetScale( pcbView->GetScale() * ( scale ) );
2492
2493 //Let's refocus because there is an algorithm to avoid dialogs in there.
2494 m_frame->FocusOnLocation( aItem->GetCenter() );
2495 }
2496 }
2497 }
2498 // Inform other potentially interested tools
2500 }
2501 else if( cleared )
2502 {
2504 }
2505
2507}
2508
2509
2515static bool itemIsIncludedByFilter( const BOARD_ITEM& aItem, const BOARD& aBoard,
2516 const DIALOG_FILTER_SELECTION::OPTIONS& aFilterOptions )
2517{
2518 switch( aItem.Type() )
2519 {
2520 case PCB_FOOTPRINT_T:
2521 {
2522 const FOOTPRINT& footprint = static_cast<const FOOTPRINT&>( aItem );
2523
2524 return aFilterOptions.includeModules
2525 && ( aFilterOptions.includeLockedModules || !footprint.IsLocked() );
2526 }
2527
2528 case PCB_TRACE_T:
2529 case PCB_ARC_T:
2530 return aFilterOptions.includeTracks;
2531
2532 case PCB_VIA_T:
2533 return aFilterOptions.includeVias;
2534
2535 case PCB_ZONE_T:
2536 return aFilterOptions.includeZones;
2537
2538 case PCB_SHAPE_T:
2539 case PCB_TARGET_T:
2540 case PCB_DIM_ALIGNED_T:
2541 case PCB_DIM_CENTER_T:
2542 case PCB_DIM_RADIAL_T:
2544 case PCB_DIM_LEADER_T:
2545 if( aItem.GetLayer() == Edge_Cuts )
2546 return aFilterOptions.includeBoardOutlineLayer;
2547 else
2548 return aFilterOptions.includeItemsOnTechLayers;
2549
2550 case PCB_FIELD_T:
2551 case PCB_TEXT_T:
2552 case PCB_TEXTBOX_T:
2553 case PCB_TABLE_T:
2554 case PCB_TABLECELL_T:
2555 return aFilterOptions.includePcbTexts;
2556
2557 default:
2558 // Filter dialog is inclusive, not exclusive. If it's not included, then it doesn't
2559 // get selected.
2560 return false;
2561 }
2562}
2563
2564
2566{
2567 const BOARD& board = *getModel<BOARD>();
2568 DIALOG_FILTER_SELECTION::OPTIONS& opts = m_priv->m_filterOpts;
2569 DIALOG_FILTER_SELECTION dlg( m_frame, opts );
2570
2571 const int cmd = dlg.ShowModal();
2572
2573 if( cmd != wxID_OK )
2574 return 0;
2575
2576 // copy current selection
2577 std::deque<EDA_ITEM*> selection = m_selection.GetItems();
2578
2579 ClearSelection( true /*quiet mode*/ );
2580
2581 // re-select items from the saved selection according to the dialog options
2582 for( EDA_ITEM* i : selection )
2583 {
2584 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( i );
2585 bool include = itemIsIncludedByFilter( *item, board, opts );
2586
2587 if( include )
2588 select( item );
2589 }
2590
2592
2593 return 0;
2594}
2595
2596
2598{
2599 if( aCollector.GetCount() == 0 )
2600 return;
2601
2602 std::set<BOARD_ITEM*> rejected;
2603
2604 for( EDA_ITEM* i : aCollector )
2605 {
2606 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( i );
2607
2608 if( !itemPassesFilter( item, aMultiSelect ) )
2609 rejected.insert( item );
2610 }
2611
2612 for( BOARD_ITEM* item : rejected )
2613 aCollector.Remove( item );
2614}
2615
2616
2617bool PCB_SELECTION_TOOL::itemPassesFilter( BOARD_ITEM* aItem, bool aMultiSelect )
2618{
2619 if( !m_filter.lockedItems )
2620 {
2621 if( aItem->IsLocked() || ( aItem->GetParent() && aItem->GetParent()->IsLocked() ) )
2622 {
2623 if( aItem->Type() == PCB_PAD_T && !aMultiSelect )
2624 {
2625 // allow a single pad to be selected -- there are a lot of operations that
2626 // require this so we allow this one inconsistency
2627 }
2628 else
2629 {
2630 return false;
2631 }
2632 }
2633 }
2634
2635 if( !aItem )
2636 return false;
2637
2638 KICAD_T itemType = aItem->Type();
2639
2640 if( itemType == PCB_GENERATOR_T )
2641 {
2642 if( static_cast<PCB_GENERATOR*>( aItem )->GetItems().empty() )
2643 {
2644 if( !m_filter.otherItems )
2645 return false;
2646 }
2647 else
2648 {
2649 itemType = ( *static_cast<PCB_GENERATOR*>( aItem )->GetItems().begin() )->Type();
2650 }
2651 }
2652
2653 switch( itemType )
2654 {
2655 case PCB_FOOTPRINT_T:
2656 if( !m_filter.footprints )
2657 return false;
2658
2659 break;
2660
2661 case PCB_PAD_T:
2662 if( !m_filter.pads )
2663 return false;
2664
2665 break;
2666
2667 case PCB_TRACE_T:
2668 case PCB_ARC_T:
2669 if( !m_filter.tracks )
2670 return false;
2671
2672 break;
2673
2674 case PCB_VIA_T:
2675 if( !m_filter.vias )
2676 return false;
2677
2678 break;
2679
2680 case PCB_ZONE_T:
2681 {
2682 ZONE* zone = static_cast<ZONE*>( aItem );
2683
2684 if( ( !m_filter.zones && !zone->GetIsRuleArea() )
2685 || ( !m_filter.keepouts && zone->GetIsRuleArea() ) )
2686 {
2687 return false;
2688 }
2689
2690 // m_SolderMaskBridges zone is a special zone, only used to showsolder mask briges
2691 // after running DRC. it is not really a board item.
2692 // Never select it or delete by a Commit.
2693 if( zone == m_frame->GetBoard()->m_SolderMaskBridges )
2694 return false;
2695
2696 break;
2697 }
2698
2699 case PCB_SHAPE_T:
2700 case PCB_TARGET_T:
2701 if( !m_filter.graphics )
2702 return false;
2703
2704 break;
2705
2707 if( !m_filter.graphics )
2708 return false;
2709
2710 // a reference image living in a footprint must not be selected inside the board editor
2711 if( !m_isFootprintEditor && aItem->GetParentFootprint() )
2712 return false;
2713
2714 break;
2715
2716 case PCB_FIELD_T:
2717 case PCB_TEXT_T:
2718 case PCB_TEXTBOX_T:
2719 case PCB_TABLE_T:
2720 case PCB_TABLECELL_T:
2721 if( !m_filter.text )
2722 return false;
2723
2724 break;
2725
2726 case PCB_DIM_ALIGNED_T:
2727 case PCB_DIM_CENTER_T:
2728 case PCB_DIM_RADIAL_T:
2730 case PCB_DIM_LEADER_T:
2731 if( !m_filter.dimensions )
2732 return false;
2733
2734 break;
2735
2736 default:
2737 if( !m_filter.otherItems )
2738 return false;
2739 }
2740
2741 return true;
2742}
2743
2744
2746{
2747 if( m_selection.Empty() )
2748 return;
2749
2750 while( m_selection.GetSize() )
2752
2753 view()->Update( &m_selection );
2754
2755 m_selection.SetIsHover( false );
2757
2758 // Inform other potentially interested tools
2759 if( !aQuietMode )
2760 {
2763 }
2764}
2765
2766
2768{
2770
2771 bool enteredGroupFound = false;
2772
2773 INSPECTOR_FUNC inspector =
2774 [&]( EDA_ITEM* item, void* testData )
2775 {
2776 if( item->IsSelected() )
2777 {
2778 EDA_ITEM* parent = item->GetParent();
2779
2780 // Let selected parents handle their children.
2781 if( parent && parent->IsSelected() )
2782 return INSPECT_RESULT::CONTINUE;
2783
2784 highlight( item, SELECTED, &m_selection );
2785 }
2786
2787 if( item->Type() == PCB_GROUP_T )
2788 {
2789 if( item == m_enteredGroup )
2790 {
2791 item->SetFlags( ENTERED );
2792 enteredGroupFound = true;
2793 }
2794 else
2795 {
2796 item->ClearFlags( ENTERED );
2797 }
2798 }
2799
2800 return INSPECT_RESULT::CONTINUE;
2801 };
2802
2805
2806 if( !enteredGroupFound )
2807 {
2809 m_enteredGroup = nullptr;
2810 }
2811}
2812
2813
2814bool PCB_SELECTION_TOOL::Selectable( const BOARD_ITEM* aItem, bool checkVisibilityOnly ) const
2815{
2816 const RENDER_SETTINGS* settings = getView()->GetPainter()->GetSettings();
2817 const PCB_DISPLAY_OPTIONS& options = frame()->GetDisplayOptions();
2818
2819 auto visibleLayers =
2820 [&]()
2821 {
2823 {
2824 LSET set;
2825
2826 for( PCB_LAYER_ID layer : LSET::AllLayersMask().Seq() )
2827 set.set( layer, view()->IsLayerVisible( layer ) );
2828
2829 return set;
2830 }
2831 else
2832 {
2833 return board()->GetVisibleLayers();
2834 }
2835 };
2836
2837 auto layerVisible =
2838 [&]( PCB_LAYER_ID aLayer )
2839 {
2841 return view()->IsLayerVisible( aLayer );
2842 else
2843 return board()->IsLayerVisible( aLayer );
2844 };
2845
2846 if( settings->GetHighContrast() )
2847 {
2848 const std::set<int> activeLayers = settings->GetHighContrastLayers();
2849 bool onActiveLayer = false;
2850
2851 for( int layer : activeLayers )
2852 {
2853 // NOTE: Only checking the regular layers (not GAL meta-layers)
2854 if( layer < PCB_LAYER_ID_COUNT && aItem->IsOnLayer( ToLAYER_ID( layer ) ) )
2855 {
2856 onActiveLayer = true;
2857 break;
2858 }
2859 }
2860
2861 if( !onActiveLayer && aItem->Type() != PCB_MARKER_T )
2862 {
2863 // We do not want to select items that are in the background
2864 return false;
2865 }
2866 }
2867
2868 if( aItem->Type() == PCB_FOOTPRINT_T )
2869 {
2870 const FOOTPRINT* footprint = static_cast<const FOOTPRINT*>( aItem );
2871
2872 // In footprint editor, we do not want to select the footprint itself.
2874 return false;
2875
2876 // Allow selection of footprints if some part of the footprint is visible.
2877 if( footprint->GetSide() != UNDEFINED_LAYER && !m_skip_heuristics )
2878 {
2879 LSET boardSide = footprint->IsFlipped() ? LSET::BackMask() : LSET::FrontMask();
2880
2881 if( !( visibleLayers() & boardSide ).any() )
2882 return false;
2883 }
2884
2885 // If the footprint has no items except the reference and value fields, include the
2886 // footprint in the selections.
2887 if( footprint->GraphicalItems().empty()
2888 && footprint->Pads().empty()
2889 && footprint->Zones().empty() )
2890 {
2891 return true;
2892 }
2893
2894 for( const BOARD_ITEM* item : footprint->GraphicalItems() )
2895 {
2896 if( Selectable( item, true ) )
2897 return true;
2898 }
2899
2900 for( const PAD* pad : footprint->Pads() )
2901 {
2902 if( Selectable( pad, true ) )
2903 return true;
2904 }
2905
2906 for( const ZONE* zone : footprint->Zones() )
2907 {
2908 if( Selectable( zone, true ) )
2909 return true;
2910 }
2911
2912 return false;
2913 }
2914 else if( aItem->Type() == PCB_GROUP_T )
2915 {
2916 PCB_GROUP* group = const_cast<PCB_GROUP*>( static_cast<const PCB_GROUP*>( aItem ) );
2917
2918 // Similar to logic for footprint, a group is selectable if any of its members are.
2919 // (This recurses.)
2920 for( BOARD_ITEM* item : group->GetBoardItems() )
2921 {
2922 if( Selectable( item, true ) )
2923 return true;
2924 }
2925
2926 return false;
2927 }
2928
2929 if( aItem->GetParentGroup() && aItem->GetParentGroup()->AsEdaItem()->Type() == PCB_GENERATOR_T )
2930 return false;
2931
2932 const ZONE* zone = nullptr;
2933 const PCB_VIA* via = nullptr;
2934 const PAD* pad = nullptr;
2935 const PCB_TEXT* text = nullptr;
2936 const PCB_FIELD* field = nullptr;
2937
2938 // Most footprint children can only be selected in the footprint editor.
2939 if( aItem->GetParentFootprint() && !m_isFootprintEditor && !checkVisibilityOnly )
2940 {
2941 if( aItem->Type() != PCB_FIELD_T && aItem->Type() != PCB_PAD_T
2942 && aItem->Type() != PCB_TEXT_T )
2943 {
2944 return false;
2945 }
2946 }
2947
2948 switch( aItem->Type() )
2949 {
2950 case PCB_ZONE_T:
2951 if( !board()->IsElementVisible( LAYER_ZONES ) || ( options.m_ZoneOpacity == 0.00 ) )
2952 return false;
2953
2954 zone = static_cast<const ZONE*>( aItem );
2955
2956 // A teardrop is modelled as a property of a via, pad or the board (for track-to-track
2957 // teardrops). The underlying zone is only an implementation detail.
2958 if( zone->IsTeardropArea() && !board()->LegacyTeardrops() )
2959 return false;
2960
2961 // zones can exist on multiple layers!
2962 if( !( zone->GetLayerSet() & visibleLayers() ).any() )
2963 return false;
2964
2965 break;
2966
2967 case PCB_TRACE_T:
2968 case PCB_ARC_T:
2969 if( !board()->IsElementVisible( LAYER_TRACKS ) || ( options.m_TrackOpacity == 0.00 ) )
2970 return false;
2971
2972 if( !layerVisible( aItem->GetLayer() ) )
2973 return false;
2974
2975 break;
2976
2977 case PCB_VIA_T:
2978 if( !board()->IsElementVisible( LAYER_VIAS ) || ( options.m_ViaOpacity == 0.00 ) )
2979 return false;
2980
2981 via = static_cast<const PCB_VIA*>( aItem );
2982
2983 // For vias it is enough if only one of its layers is visible
2984 if( !( visibleLayers() & via->GetLayerSet() ).any() )
2985 return false;
2986
2987 break;
2988
2989 case PCB_FIELD_T:
2990 field = static_cast<const PCB_FIELD*>( aItem );
2991
2992 if( !field->IsVisible() )
2993 return false;
2994
2995 if( field->IsReference() && !view()->IsLayerVisible( LAYER_FP_REFERENCES ) )
2996 return false;
2997
2998 if( field->IsValue() && !view()->IsLayerVisible( LAYER_FP_VALUES ) )
2999 return false;
3000
3001 // Handle all other fields with normal text visibility controls
3003 case PCB_TEXT_T:
3004 text = static_cast<const PCB_TEXT*>( aItem );
3005
3006 if( !layerVisible( text->GetLayer() ) )
3007 return false;
3008
3009 // Apply the LOD visibility test as well
3010 if( !view()->IsVisible( text ) )
3011 return false;
3012
3013 if( aItem->GetParentFootprint() )
3014 {
3015 int controlLayer = LAYER_FP_TEXT;
3016
3017 if( text->GetText() == wxT( "${REFERENCE}" ) )
3018 controlLayer = LAYER_FP_REFERENCES;
3019 else if( text->GetText() == wxT( "${VALUE}" ) )
3020 controlLayer = LAYER_FP_VALUES;
3021
3022 if( !view()->IsLayerVisible( controlLayer ) )
3023 return false;
3024 }
3025
3026 break;
3027
3029 if( options.m_ImageOpacity == 0.00 )
3030 return false;
3031
3032 // Bitmap images on board are hidden if LAYER_DRAW_BITMAPS is not visible
3034 return false;
3035
3037
3038 case PCB_SHAPE_T:
3039 // Note: LAYER_SHAPES does not control the visibility of a PCB_SHAPE_T, only
3040 // the opacity of filled areas
3041 // The visibility is managed by the item layer
3042 if( options.m_FilledShapeOpacity == 0.0 )
3043 return false;
3044
3046
3047 case PCB_TEXTBOX_T:
3048 case PCB_TABLE_T:
3049 case PCB_TABLECELL_T:
3050 if( !layerVisible( aItem->GetLayer() ) )
3051 return false;
3052
3053 if( aItem->Type() == PCB_TABLECELL_T )
3054 {
3055 const PCB_TABLECELL* cell = static_cast<const PCB_TABLECELL*>( aItem );
3056
3057 if( cell->GetRowSpan() == 0 || cell->GetColSpan() == 0 )
3058 return false;
3059 }
3060
3061 break;
3062
3063 case PCB_DIM_ALIGNED_T:
3064 case PCB_DIM_LEADER_T:
3065 case PCB_DIM_CENTER_T:
3066 case PCB_DIM_RADIAL_T:
3068 if( !layerVisible( aItem->GetLayer() ) )
3069 return false;
3070
3071 break;
3072
3073 case PCB_PAD_T:
3074 if( options.m_PadOpacity == 0.00 )
3075 return false;
3076
3077 pad = static_cast<const PAD*>( aItem );
3078
3079 if( pad->GetAttribute() == PAD_ATTRIB::PTH || pad->GetAttribute() == PAD_ATTRIB::NPTH )
3080 {
3081 // A pad's hole is visible on every layer the pad is visible on plus many layers the
3082 // pad is not visible on -- so we only need to check for any visible hole layers.
3083 if( !( visibleLayers() & LSET::PhysicalLayersMask() ).any() )
3084 return false;
3085 }
3086 else
3087 {
3088 if( !( pad->GetLayerSet() & visibleLayers() ).any() )
3089 return false;
3090 }
3091
3092 break;
3093
3094 // These are not selectable
3095 case PCB_NETINFO_T:
3096 case NOT_USED:
3097 case TYPE_NOT_INIT:
3098 return false;
3099
3100 default: // Suppress warnings
3101 break;
3102 }
3103
3104 return true;
3105}
3106
3107
3109{
3110 if( !aItem || aItem->IsSelected() )
3111 return;
3112
3113 if( aItem->Type() == PCB_PAD_T )
3114 {
3115 FOOTPRINT* footprint = static_cast<FOOTPRINT*>( aItem->GetParent() );
3116
3117 if( m_selection.Contains( footprint ) )
3118 return;
3119 }
3120
3121 if( m_enteredGroup &&
3122 !PCB_GROUP::WithinScope( static_cast<BOARD_ITEM*>( aItem ), m_enteredGroup,
3124 {
3125 ExitGroup();
3126 }
3127
3128 highlight( aItem, SELECTED, &m_selection );
3129}
3130
3131
3133{
3134 unhighlight( aItem, SELECTED, &m_selection );
3135}
3136
3137
3138void PCB_SELECTION_TOOL::highlight( EDA_ITEM* aItem, int aMode, SELECTION* aGroup )
3139{
3140 if( aGroup )
3141 aGroup->Add( aItem );
3142
3143 highlightInternal( aItem, aMode, aGroup != nullptr );
3144 view()->Update( aItem, KIGFX::REPAINT );
3145
3146 // Many selections are very temporal and updating the display each time just
3147 // creates noise.
3148 if( aMode == BRIGHTENED )
3150}
3151
3152
3153void PCB_SELECTION_TOOL::highlightInternal( EDA_ITEM* aItem, int aMode, bool aUsingOverlay )
3154{
3155 if( aMode == SELECTED )
3156 aItem->SetSelected();
3157 else if( aMode == BRIGHTENED )
3158 aItem->SetBrightened();
3159
3160 if( aUsingOverlay && aMode != BRIGHTENED )
3161 view()->Hide( aItem, true ); // Hide the original item, so it is shown only on overlay
3162
3163 if( aItem->IsBOARD_ITEM() )
3164 {
3165 BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( aItem );
3166 boardItem->RunOnChildren( std::bind( &PCB_SELECTION_TOOL::highlightInternal, this, _1, aMode, aUsingOverlay ),
3167 RECURSE_MODE::RECURSE );
3168 }
3169}
3170
3171
3172void PCB_SELECTION_TOOL::unhighlight( EDA_ITEM* aItem, int aMode, SELECTION* aGroup )
3173{
3174 if( aGroup )
3175 aGroup->Remove( aItem );
3176
3177 unhighlightInternal( aItem, aMode, aGroup != nullptr );
3178 view()->Update( aItem, KIGFX::REPAINT );
3179
3180 // Many selections are very temporal and updating the display each time just creates noise.
3181 if( aMode == BRIGHTENED )
3183}
3184
3185
3186void PCB_SELECTION_TOOL::unhighlightInternal( EDA_ITEM* aItem, int aMode, bool aUsingOverlay )
3187{
3188 if( aMode == SELECTED )
3189 aItem->ClearSelected();
3190 else if( aMode == BRIGHTENED )
3191 aItem->ClearBrightened();
3192
3193 if( aUsingOverlay && aMode != BRIGHTENED )
3194 {
3195 view()->Hide( aItem, false ); // Restore original item visibility...
3196 view()->Update( aItem ); // ... and make sure it's redrawn un-selected
3197 }
3198
3199 if( aItem->IsBOARD_ITEM() )
3200 {
3201 BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( aItem );
3202 boardItem->RunOnChildren( std::bind( &PCB_SELECTION_TOOL::unhighlightInternal, this, _1, aMode, aUsingOverlay ),
3203 RECURSE_MODE::RECURSE );
3204 }
3205}
3206
3207
3209{
3210 const unsigned GRIP_MARGIN = 20;
3211 int margin = KiROUND( getView()->ToWorld( GRIP_MARGIN ) );
3212
3213 // Check if the point is located close to any of the currently selected items
3214 for( EDA_ITEM* item : m_selection )
3215 {
3216 BOX2I itemBox = item->ViewBBox();
3217 itemBox.Inflate( margin ); // Give some margin for gripping an item
3218
3219 if( itemBox.Contains( aPoint ) )
3220 {
3221 if( item->HitTest( aPoint, margin ) )
3222 return true;
3223
3224 bool found = false;
3225
3226 if( PCB_GROUP* group = dynamic_cast<PCB_GROUP*>( item ) )
3227 {
3228 group->RunOnChildren(
3229 [&]( BOARD_ITEM* aItem )
3230 {
3231 if( aItem->HitTest( aPoint, margin ) )
3232 found = true;
3233 },
3234 RECURSE_MODE::RECURSE );
3235 }
3236
3237 if( found )
3238 return true;
3239 }
3240 }
3241
3242 return false;
3243}
3244
3245
3247 int aMaxDistance ) const
3248{
3249 BOX2D viewportD = getView()->GetViewport();
3250 BOX2I viewport = BOX2ISafe( viewportD );
3251 int distance = INT_MAX;
3252 SEG loc( aWhere, aWhere );
3253
3254 switch( aItem->Type() )
3255 {
3256 case PCB_FIELD_T:
3257 case PCB_TEXT_T:
3258 {
3259 PCB_TEXT* text = static_cast<PCB_TEXT*>( aItem );
3260
3261 // Add a bit of slop to text-shapes
3262 if( text->GetEffectiveTextShape()->Collide( loc, aMaxDistance, &distance ) )
3263 distance = std::clamp( distance - ( aMaxDistance / 2 ), 0, distance );
3264
3265 break;
3266 }
3267
3268 case PCB_TEXTBOX_T:
3269 case PCB_TABLECELL_T:
3270 {
3271 PCB_TEXTBOX* textbox = static_cast<PCB_TEXTBOX*>( aItem );
3272
3273 // Add a bit of slop to text-shapes
3274 if( textbox->GetEffectiveTextShape()->Collide( loc, aMaxDistance, &distance ) )
3275 distance = std::clamp( distance - ( aMaxDistance / 2 ), 0, distance );
3276
3277 break;
3278 }
3279
3280 case PCB_TABLE_T:
3281 {
3282 PCB_TABLE* table = static_cast<PCB_TABLE*>( aItem );
3283
3284 for( PCB_TABLECELL* cell : table->GetCells() )
3285 {
3286 // Add a bit of slop to text-shapes
3287 if( cell->GetEffectiveTextShape()->Collide( loc, aMaxDistance, &distance ) )
3288 distance = std::clamp( distance - ( aMaxDistance / 2 ), 0, distance );
3289 }
3290
3291 break;
3292 }
3293
3294 case PCB_ZONE_T:
3295 {
3296 ZONE* zone = static_cast<ZONE*>( aItem );
3297
3298 // Zone borders are very specific
3299 if( zone->HitTestForEdge( aWhere, aMaxDistance / 2 ) )
3300 distance = 0;
3301 else if( zone->HitTestForEdge( aWhere, aMaxDistance ) )
3302 distance = aMaxDistance / 2;
3303 else
3304 aItem->GetEffectiveShape()->Collide( loc, aMaxDistance, &distance );
3305
3306 break;
3307 }
3308
3309 case PCB_FOOTPRINT_T:
3310 {
3311 FOOTPRINT* footprint = static_cast<FOOTPRINT*>( aItem );
3312 BOX2I bbox = footprint->GetBoundingBox( false );
3313
3314 try
3315 {
3316 footprint->GetBoundingHull().Collide( loc, aMaxDistance, &distance );
3317 }
3318 catch( const std::exception& e )
3319 {
3320 wxFAIL_MSG( wxString::Format( wxT( "Clipper exception occurred: %s" ), e.what() ) );
3321 }
3322
3323 // Consider footprints larger than the viewport only as a last resort
3324 if( bbox.GetHeight() > viewport.GetHeight() || bbox.GetWidth() > viewport.GetWidth() )
3325 distance = INT_MAX / 2;
3326
3327 break;
3328 }
3329
3330 case PCB_MARKER_T:
3331 {
3332 PCB_MARKER* marker = static_cast<PCB_MARKER*>( aItem );
3333 SHAPE_LINE_CHAIN polygon;
3334
3335 marker->ShapeToPolygon( polygon );
3336 polygon.Move( marker->GetPos() );
3337 polygon.Collide( loc, aMaxDistance, &distance );
3338 break;
3339 }
3340
3341 case PCB_GROUP_T:
3342 case PCB_GENERATOR_T:
3343 {
3344 PCB_GROUP* group = static_cast<PCB_GROUP*>( aItem );
3345
3346 for( BOARD_ITEM* member : group->GetBoardItems() )
3347 distance = std::min( distance, hitTestDistance( aWhere, member, aMaxDistance ) );
3348
3349 break;
3350 }
3351
3352 case PCB_PAD_T:
3353 {
3354 static_cast<PAD*>( aItem )->Padstack().ForEachUniqueLayer(
3355 [&]( PCB_LAYER_ID aLayer )
3356 {
3357 int layerDistance = INT_MAX;
3358 aItem->GetEffectiveShape( aLayer )->Collide( loc, aMaxDistance, &layerDistance );
3359 distance = std::min( distance, layerDistance );
3360 } );
3361
3362 break;
3363 }
3364
3365 default:
3366 aItem->GetEffectiveShape()->Collide( loc, aMaxDistance, &distance );
3367 break;
3368 }
3369
3370 return distance;
3371}
3372
3373
3375{
3376 wxCHECK( m_frame, /* void */ );
3377
3378 if( aCollector.GetCount() < 2 )
3379 return;
3380
3381 const RENDER_SETTINGS* settings = getView()->GetPainter()->GetSettings();
3382
3383 wxCHECK( settings, /* void */ );
3384
3385 PCB_LAYER_ID activeLayer = m_frame->GetActiveLayer();
3386 LSET visibleLayers = m_frame->GetBoard()->GetVisibleLayers();
3387 LSET enabledLayers = m_frame->GetBoard()->GetEnabledLayers();
3388 LSEQ enabledLayerStack = enabledLayers.SeqStackupTop2Bottom( activeLayer );
3389
3390 wxCHECK( !enabledLayerStack.empty(), /* void */ );
3391
3392 auto isZoneFillKeepout =
3393 []( const BOARD_ITEM* aItem ) -> bool
3394 {
3395 if( aItem->Type() == PCB_ZONE_T )
3396 {
3397 const ZONE* zone = static_cast<const ZONE*>( aItem );
3398
3399 if( zone->GetIsRuleArea() && zone->GetDoNotAllowZoneFills() )
3400 return true;
3401 }
3402
3403 return false;
3404 };
3405
3406 std::vector<LAYER_OPACITY_ITEM> opacityStackup;
3407
3408 for( int i = 0; i < aCollector.GetCount(); i++ )
3409 {
3410 const BOARD_ITEM* item = aCollector[i];
3411
3412 LSET itemLayers = item->GetLayerSet() & enabledLayers & visibleLayers;
3413 LSEQ itemLayerSeq = itemLayers.Seq( enabledLayerStack );
3414
3415 for( PCB_LAYER_ID layer : itemLayerSeq )
3416 {
3417 COLOR4D color = settings->GetColor( item, layer );
3418
3419 if( color.a == 0 )
3420 continue;
3421
3422 LAYER_OPACITY_ITEM opacityItem;
3423
3424 opacityItem.m_Layer = layer;
3425 opacityItem.m_Opacity = color.a;
3426 opacityItem.m_Item = item;
3427
3428 if( isZoneFillKeepout( item ) )
3429 opacityItem.m_Opacity = 0.0;
3430
3431 opacityStackup.emplace_back( opacityItem );
3432 }
3433 }
3434
3435 std::sort( opacityStackup.begin(), opacityStackup.end(),
3436 [&]( const LAYER_OPACITY_ITEM& aLhs, const LAYER_OPACITY_ITEM& aRhs ) -> bool
3437 {
3438 int retv = enabledLayerStack.TestLayers( aLhs.m_Layer, aRhs.m_Layer );
3439
3440 if( retv )
3441 return retv > 0;
3442
3443 return aLhs.m_Opacity > aRhs.m_Opacity;
3444 } );
3445
3446 std::set<const BOARD_ITEM*> visibleItems;
3447 std::set<const BOARD_ITEM*> itemsToRemove;
3448 double minAlphaLimit = ADVANCED_CFG::GetCfg().m_PcbSelectionVisibilityRatio;
3449 double currentStackupOpacity = 0.0;
3450 PCB_LAYER_ID lastVisibleLayer = PCB_LAYER_ID::UNDEFINED_LAYER;
3451
3452 for( const LAYER_OPACITY_ITEM& opacityItem : opacityStackup )
3453 {
3454 if( lastVisibleLayer == PCB_LAYER_ID::UNDEFINED_LAYER )
3455 {
3456 currentStackupOpacity = opacityItem.m_Opacity;
3457 lastVisibleLayer = opacityItem.m_Layer;
3458 visibleItems.emplace( opacityItem.m_Item );
3459 continue;
3460 }
3461
3462 // Objects to ignore and fallback to the old selection behavior.
3463 auto ignoreItem =
3464 [&]()
3465 {
3466 const BOARD_ITEM* item = opacityItem.m_Item;
3467
3468 wxCHECK( item, false );
3469
3470 // Check items that span multiple layers for visibility.
3471 if( visibleItems.count( item ) )
3472 return true;
3473
3474 // Don't prune child items of a footprint that is already visible.
3475 if( item->GetParent()
3476 && ( item->GetParent()->Type() == PCB_FOOTPRINT_T )
3477 && visibleItems.count( item->GetParent() ) )
3478 {
3479 return true;
3480 }
3481
3482 // Keepout zones are transparent but for some reason, PCB_PAINTER::GetColor()
3483 // returns the color of the zone it prevents from filling.
3484 if( isZoneFillKeepout( item ) )
3485 return true;
3486
3487 return false;
3488 };
3489
3490 // Everything on the currently selected layer is visible;
3491 if( opacityItem.m_Layer == enabledLayerStack[0] )
3492 {
3493 visibleItems.emplace( opacityItem.m_Item );
3494 }
3495 else
3496 {
3497 double itemVisibility = opacityItem.m_Opacity * ( 1.0 - currentStackupOpacity );
3498
3499 if( ( itemVisibility <= minAlphaLimit ) && !ignoreItem() )
3500 itemsToRemove.emplace( opacityItem.m_Item );
3501 else
3502 visibleItems.emplace( opacityItem.m_Item );
3503 }
3504
3505 if( opacityItem.m_Layer != lastVisibleLayer )
3506 {
3507 currentStackupOpacity += opacityItem.m_Opacity * ( 1.0 - currentStackupOpacity );
3508 currentStackupOpacity = std::min( currentStackupOpacity, 1.0 );
3509 lastVisibleLayer = opacityItem.m_Layer;
3510 }
3511 }
3512
3513 for( const BOARD_ITEM* itemToRemove : itemsToRemove )
3514 {
3515 wxCHECK( aCollector.GetCount() > 1, /* void */ );
3516 aCollector.Remove( itemToRemove );
3517 }
3518}
3519
3520
3521// The general idea here is that if the user clicks directly on a small item inside a larger
3522// one, then they want the small item. The quintessential case of this is clicking on a pad
3523// within a footprint, but we also apply it for text within a footprint, footprints within
3524// larger footprints, and vias within either larger pads or longer tracks.
3525//
3526// These "guesses" presume there is area within the larger item to click in to select it. If
3527// an item is mostly covered by smaller items within it, then the guesses are inappropriate as
3528// there might not be any area left to click to select the larger item. In this case we must
3529// leave the items in the collector and bring up a Selection Clarification menu.
3530//
3531// We currently check for pads and text mostly covering a footprint, but we don't check for
3532// smaller footprints mostly covering a larger footprint.
3533//
3535 const VECTOR2I& aWhere ) const
3536{
3537 static const LSET silkLayers( { B_SilkS, F_SilkS } );
3538 static const LSET courtyardLayers( { B_CrtYd, F_CrtYd } );
3539 static std::vector<KICAD_T> singleLayerSilkTypes = { PCB_FIELD_T,
3542 PCB_SHAPE_T };
3543
3544 if( ADVANCED_CFG::GetCfg().m_PcbSelectionVisibilityRatio != 1.0 )
3546
3547 if( aCollector.GetCount() == 1 )
3548 return;
3549
3550 std::set<BOARD_ITEM*> preferred;
3551 std::set<BOARD_ITEM*> rejected;
3552 VECTOR2I where( aWhere.x, aWhere.y );
3553 const RENDER_SETTINGS* settings = getView()->GetPainter()->GetSettings();
3554 PCB_LAYER_ID activeLayer = m_frame->GetActiveLayer();
3555
3556 // If a silk layer is in front, we assume the user is working with silk and give preferential
3557 // treatment to single-layer items on *either* silk layer.
3558 if( silkLayers[activeLayer] )
3559 {
3560 for( int i = 0; i < aCollector.GetCount(); ++i )
3561 {
3562 BOARD_ITEM* item = aCollector[i];
3563
3564 if( item->IsType( singleLayerSilkTypes ) && silkLayers[ item->GetLayer() ] )
3565 preferred.insert( item );
3566 }
3567 }
3568 // Similarly, if a courtyard layer is in front, we assume the user is positioning footprints
3569 // and give preferential treatment to footprints on *both* top and bottom.
3570 else if( courtyardLayers[activeLayer] && settings->GetHighContrast() )
3571 {
3572 for( int i = 0; i < aCollector.GetCount(); ++i )
3573 {
3574 BOARD_ITEM* item = aCollector[i];
3575
3576 if( item->Type() == PCB_FOOTPRINT_T )
3577 preferred.insert( item );
3578 }
3579 }
3580
3581 if( preferred.size() > 0 )
3582 {
3583 aCollector.Empty();
3584
3585 for( BOARD_ITEM* item : preferred )
3586 aCollector.Append( item );
3587
3588 if( preferred.size() == 1 )
3589 return;
3590 }
3591
3592 // Prefer exact hits to sloppy ones
3593 constexpr int MAX_SLOP = 5;
3594
3595 int singlePixel = KiROUND( aCollector.GetGuide()->OnePixelInIU() );
3596 int maxSlop = KiROUND( MAX_SLOP * aCollector.GetGuide()->OnePixelInIU() );
3597 int minSlop = INT_MAX;
3598
3599 std::map<BOARD_ITEM*, int> itemsBySloppiness;
3600
3601 for( int i = 0; i < aCollector.GetCount(); ++i )
3602 {
3603 BOARD_ITEM* item = aCollector[i];
3604 int itemSlop = hitTestDistance( where, item, maxSlop );
3605
3606 itemsBySloppiness[ item ] = itemSlop;
3607
3608 if( itemSlop < minSlop )
3609 minSlop = itemSlop;
3610 }
3611
3612 // Prune sloppier items
3613 if( minSlop < INT_MAX )
3614 {
3615 for( std::pair<BOARD_ITEM*, int> pair : itemsBySloppiness )
3616 {
3617 if( pair.second > minSlop + singlePixel )
3618 aCollector.Transfer( pair.first );
3619 }
3620 }
3621
3622 // If the user clicked on a small item within a much larger one then it's pretty clear
3623 // they're trying to select the smaller one.
3624 constexpr double sizeRatio = 1.5;
3625
3626 std::vector<std::pair<BOARD_ITEM*, double>> itemsByArea;
3627
3628 for( int i = 0; i < aCollector.GetCount(); ++i )
3629 {
3630 BOARD_ITEM* item = aCollector[i];
3631 double area = 0.0;
3632
3633 if( item->Type() == PCB_ZONE_T
3634 && static_cast<ZONE*>( item )->HitTestForEdge( where, maxSlop / 2 ) )
3635 {
3636 // Zone borders are very specific, so make them "small"
3637 area = (double) SEG::Square( singlePixel ) * MAX_SLOP;
3638 }
3639 else if( item->Type() == PCB_VIA_T )
3640 {
3641 // Vias rarely hide other things, and we don't want them deferring to short track
3642 // segments underneath them -- so artificially reduce their size from πr² to r².
3643 area = (double) SEG::Square( static_cast<PCB_VIA*>( item )->GetDrill() / 2 );
3644 }
3645 else if( item->Type() == PCB_REFERENCE_IMAGE_T )
3646 {
3647 BOX2I box = item->GetBoundingBox();
3648 area = (double) box.GetWidth() * box.GetHeight();
3649 }
3650 else
3651 {
3652 try
3653 {
3654 area = FOOTPRINT::GetCoverageArea( item, aCollector );
3655 }
3656 catch( const std::exception& e )
3657 {
3658 wxFAIL_MSG( wxString::Format( wxT( "Clipper exception occurred: %s" ), e.what() ) );
3659 }
3660 }
3661
3662 itemsByArea.emplace_back( item, area );
3663 }
3664
3665 std::sort( itemsByArea.begin(), itemsByArea.end(),
3666 []( const std::pair<BOARD_ITEM*, double>& lhs,
3667 const std::pair<BOARD_ITEM*, double>& rhs ) -> bool
3668 {
3669 return lhs.second < rhs.second;
3670 } );
3671
3672 bool rejecting = false;
3673
3674 for( int i = 1; i < (int) itemsByArea.size(); ++i )
3675 {
3676 if( itemsByArea[i].second > itemsByArea[i-1].second * sizeRatio )
3677 rejecting = true;
3678
3679 if( rejecting )
3680 rejected.insert( itemsByArea[i].first );
3681 }
3682
3683 // Special case: if a footprint is completely covered with other features then there's no
3684 // way to select it -- so we need to leave it in the list for user disambiguation.
3685 constexpr double maxCoverRatio = 0.70;
3686
3687 for( int i = 0; i < aCollector.GetCount(); ++i )
3688 {
3689 if( FOOTPRINT* footprint = dynamic_cast<FOOTPRINT*>( aCollector[i] ) )
3690 {
3691 if( footprint->CoverageRatio( aCollector ) > maxCoverRatio )
3692 rejected.erase( footprint );
3693 }
3694 }
3695
3696 // Hopefully we've now got what the user wanted.
3697 if( (unsigned) aCollector.GetCount() > rejected.size() ) // do not remove everything
3698 {
3699 for( BOARD_ITEM* item : rejected )
3700 aCollector.Transfer( item );
3701 }
3702
3703 // Finally, what we are left with is a set of items of similar coverage area. We now reject
3704 // any that are not on the active layer, to reduce the number of disambiguation menus shown.
3705 // If the user wants to force-disambiguate, they can either switch layers or use the modifier
3706 // key to force the menu.
3707 if( aCollector.GetCount() > 1 )
3708 {
3709 bool haveItemOnActive = false;
3710 rejected.clear();
3711
3712 for( int i = 0; i < aCollector.GetCount(); ++i )
3713 {
3714 if( !aCollector[i]->IsOnLayer( activeLayer ) )
3715 rejected.insert( aCollector[i] );
3716 else
3717 haveItemOnActive = true;
3718 }
3719
3720 if( haveItemOnActive )
3721 {
3722 for( BOARD_ITEM* item : rejected )
3723 aCollector.Transfer( item );
3724 }
3725 }
3726}
3727
3728
3730 bool aMultiselect ) const
3731{
3732 std::unordered_set<EDA_ITEM*> toAdd;
3733
3734 // Set CANDIDATE on all parents which are included in the GENERAL_COLLECTOR. This
3735 // algorithm is O(3n), whereas checking for the parent inclusion could potentially be O(n^2).
3736 for( int j = 0; j < aCollector.GetCount(); j++ )
3737 {
3738 if( aCollector[j]->GetParent() )
3739 aCollector[j]->GetParent()->ClearFlags( CANDIDATE );
3740
3741 if( aCollector[j]->GetParentFootprint() )
3742 aCollector[j]->GetParentFootprint()->ClearFlags( CANDIDATE );
3743 }
3744
3745 if( aMultiselect )
3746 {
3747 for( int j = 0; j < aCollector.GetCount(); j++ )
3748 aCollector[j]->SetFlags( CANDIDATE );
3749 }
3750
3751 for( int j = 0; j < aCollector.GetCount(); )
3752 {
3753 BOARD_ITEM* item = aCollector[j];
3754 FOOTPRINT* fp = item->GetParentFootprint();
3755 BOARD_ITEM* start = item;
3756
3757 if( !m_isFootprintEditor && fp )
3758 start = fp;
3759
3760 // If a group is entered, disallow selections of objects outside the group.
3762 {
3763 aCollector.Remove( item );
3764 continue;
3765 }
3766
3767 // If any element is a member of a group, replace those elements with the top containing
3768 // group.
3770 {
3771 if( top->AsEdaItem() != item )
3772 {
3773 toAdd.insert( top->AsEdaItem() );
3774 top->AsEdaItem()->SetFlags( CANDIDATE );
3775
3776 aCollector.Remove( item );
3777 continue;
3778 }
3779 }
3780
3781 // Footprints are a bit easier as they can't be nested.
3782 if( fp && ( fp->GetFlags() & CANDIDATE ) )
3783 {
3784 // Remove children of selected items
3785 aCollector.Remove( item );
3786 continue;
3787 }
3788
3789 ++j;
3790 }
3791
3792 for( EDA_ITEM* item : toAdd )
3793 {
3794 if( !aCollector.HasItem( item ) )
3795 aCollector.Append( item );
3796 }
3797}
3798
3799
3801{
3802 std::set<BOARD_ITEM*> to_add;
3803
3804 // Iterate from the back so we don't have to worry about removals.
3805 for( int i = (int) aCollector.GetCount() - 1; i >= 0; --i )
3806 {
3807 BOARD_ITEM* item = aCollector[i];
3808
3809 if( item->Type() == PCB_TABLECELL_T )
3810 {
3811 if( !aCollector.HasItem( item->GetParent() ) )
3812 to_add.insert( item->GetParent() );
3813
3814 aCollector.Remove( item );
3815 }
3816 }
3817
3818 for( BOARD_ITEM* item : to_add )
3819 aCollector.Append( item );
3820}
3821
3822
3824 bool aForcePromotion ) const
3825{
3826 std::set<BOARD_ITEM*> to_add;
3827
3828 // Iterate from the back so we don't have to worry about removals.
3829 for( int i = aCollector.GetCount() - 1; i >= 0; --i )
3830 {
3831 BOARD_ITEM* item = aCollector[i];
3832
3833 if( !m_isFootprintEditor && item->Type() == PCB_PAD_T
3834 && ( !frame()->GetPcbNewSettings()->m_AllowFreePads || aForcePromotion ) )
3835 {
3836 if( !aCollector.HasItem( item->GetParent() ) )
3837 to_add.insert( item->GetParent() );
3838
3839 aCollector.Remove( item );
3840 }
3841 }
3842
3843 for( BOARD_ITEM* item : to_add )
3844 aCollector.Append( item );
3845}
3846
3847
3849{
3850 // Iterate from the back so we don't have to worry about removals.
3851 for( int i = aCollector.GetCount() - 1; i >= 0; --i )
3852 {
3853 BOARD_ITEM* item = aCollector[i];
3854
3855 if( item->Type() == PCB_MARKER_T )
3856 aCollector.Remove( item );
3857 }
3858}
3859
3860
3862 const VECTOR2I& aWhere ) const
3863{
3864 const RENDER_SETTINGS* settings = getView()->GetPainter()->GetSettings();
3865 BOX2D viewport = getView()->GetViewport();
3866 BOX2I extents = BOX2ISafe( viewport );
3867
3868 bool need_direct_hit = false;
3869 FOOTPRINT* single_fp = nullptr;
3870
3871 // If the designer is not modifying the existing selection AND we already have
3872 // a selection, then we only want to select items that are directly under the cursor.
3873 // This prevents us from being unable to clear the selection when zoomed into a footprint
3875 {
3876 need_direct_hit = true;
3877
3878 for( EDA_ITEM* item : m_selection )
3879 {
3880 FOOTPRINT* fp = nullptr;
3881
3882 if( item->Type() != PCB_FOOTPRINT_T )
3883 fp = static_cast<BOARD_ITEM*>( item )->GetParentFootprint();
3884 else
3885 fp = static_cast<FOOTPRINT*>( item );
3886
3887 // If the selection contains items that are not footprints, then don't restrict
3888 // whether we deselect the item or not.
3889 if( !fp )
3890 {
3891 single_fp = nullptr;
3892 break;
3893 }
3894 else if( !single_fp )
3895 {
3896 single_fp = fp;
3897 }
3898 // If the selection contains items from multiple footprints, then don't restrict
3899 // whether we deselect the item or not.
3900 else if( single_fp != fp )
3901 {
3902 single_fp = nullptr;
3903 break;
3904 }
3905 }
3906 }
3907
3908 auto visibleLayers =
3909 [&]()
3910 {
3912 {
3913 LSET set;
3914
3915 for( PCB_LAYER_ID layer : LSET::AllLayersMask().Seq() )
3916 set.set( layer, view()->IsLayerVisible( layer ) );
3917
3918 return set;
3919 }
3920 else
3921 {
3922 return board()->GetVisibleLayers();
3923 }
3924 };
3925
3926 LSET layers = visibleLayers();
3927
3928 if( settings->GetHighContrast() )
3929 {
3930 layers.reset();
3931
3932 const std::set<int> activeLayers = settings->GetHighContrastLayers();
3933
3934 for( int layer : activeLayers )
3935 {
3936 if( layer >= 0 && layer < PCB_LAYER_ID_COUNT )
3937 layers.set( layer );
3938 }
3939 }
3940
3941 // Iterate from the back so we don't have to worry about removals.
3942 for( int i = aCollector.GetCount() - 1; i >= 0; --i )
3943 {
3944 BOARD_ITEM* item = aCollector[i];
3945 FOOTPRINT* fp = dyn_cast<FOOTPRINT*>( item );
3946
3947 if( !fp )
3948 continue;
3949
3950 // Make footprints not difficult to select in high-contrast modes.
3951 if( layers[fp->GetLayer()] )
3952 continue;
3953
3954 BOX2I bbox = fp->GetLayerBoundingBox( layers );
3955
3956 // If the point clicked is not inside the visible bounding box, we can also remove it.
3957 if( !bbox.Contains( aWhere) )
3958 aCollector.Remove( item );
3959
3960 bool has_hit = false;
3961
3962 for( PCB_LAYER_ID layer : layers.Seq() )
3963 {
3964 if( fp->HitTestOnLayer( extents, false, layer ) )
3965 {
3966 has_hit = true;
3967 break;
3968 }
3969 }
3970
3971 // If the point is outside of the visible bounding box, we can remove it.
3972 if( !has_hit )
3973 {
3974 aCollector.Remove( item );
3975 }
3976 // Do not require a direct hit on this fp if the existing selection only contains
3977 // this fp's items. This allows you to have a selection of pads from a single
3978 // footprint and still click in the center of the footprint to select it.
3979 else if( single_fp )
3980 {
3981 if( fp == single_fp )
3982 continue;
3983 }
3984 else if( need_direct_hit )
3985 {
3986 has_hit = false;
3987
3988 for( PCB_LAYER_ID layer : layers.Seq() )
3989 {
3990 if( fp->HitTestOnLayer( aWhere, layer ) )
3991 {
3992 has_hit = true;
3993 break;
3994 }
3995 }
3996
3997 if( !has_hit )
3998 aCollector.Remove( item );
3999 }
4000 }
4001}
4002
4003
4005{
4006 getView()->Update( &m_selection );
4008
4009 return 0;
4010}
4011
4012
4014{
4015 std::set<std::pair<PCB_TABLE*, int>> columns;
4016 bool added = false;
4017
4018 for( EDA_ITEM* item : m_selection )
4019 {
4020 if( PCB_TABLECELL* cell = dynamic_cast<PCB_TABLECELL*>( item ) )
4021 {
4022 PCB_TABLE* table = static_cast<PCB_TABLE*>( cell->GetParent() );
4023 columns.insert( std::make_pair( table, cell->GetColumn() ) );
4024 }
4025 }
4026
4027 for( auto& [ table, col ] : columns )
4028 {
4029 for( int row = 0; row < table->GetRowCount(); ++row )
4030 {
4031 PCB_TABLECELL* cell = table->GetCell( row, col );
4032
4033 if( !cell->IsSelected() )
4034 {
4035 select( table->GetCell( row, col ) );
4036 added = true;
4037 }
4038 }
4039 }
4040
4041 if( added )
4043
4044 return 0;
4045}
4046
4047
4049{
4050 std::set<std::pair<PCB_TABLE*, int>> rows;
4051 bool added = false;
4052
4053 for( EDA_ITEM* item : m_selection )
4054 {
4055 if( PCB_TABLECELL* cell = dynamic_cast<PCB_TABLECELL*>( item ) )
4056 {
4057 PCB_TABLE* table = static_cast<PCB_TABLE*>( cell->GetParent() );
4058 rows.insert( std::make_pair( table, cell->GetRow() ) );
4059 }
4060 }
4061
4062 for( auto& [ table, row ] : rows )
4063 {
4064 for( int col = 0; col < table->GetColCount(); ++col )
4065 {
4066 PCB_TABLECELL* cell = table->GetCell( row, col );
4067
4068 if( !cell->IsSelected() )
4069 {
4070 select( table->GetCell( row, col ) );
4071 added = true;
4072 }
4073 }
4074 }
4075
4076 if( added )
4078
4079 return 0;
4080}
4081
4082
4084{
4085 std::set<PCB_TABLE*> tables;
4086 bool added = false;
4087
4088 for( EDA_ITEM* item : m_selection )
4089 {
4090 if( PCB_TABLECELL* cell = dynamic_cast<PCB_TABLECELL*>( item ) )
4091 tables.insert( static_cast<PCB_TABLE*>( cell->GetParent() ) );
4092 }
4093
4095
4096 for( PCB_TABLE* table : tables )
4097 {
4098 if( !table->IsSelected() )
4099 {
4100 select( table );
4101 added = true;
4102 }
4103 }
4104
4105 if( added )
4107
4108 return 0;
4109}
4110
4111
4113{
4115
4119
4126
4145
4148
4150}
int color
Definition: DXF_plotter.cpp:60
std::function< void(const VECTOR2I &, GENERAL_COLLECTOR &, PCB_SELECTION_TOOL *)> CLIENT_SELECTION_FILTER
Definition: actions.h:38
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
constexpr BOX2I BOX2ISafe(const BOX2D &aInput)
Definition: box2.h:929
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
static TOOL_ACTION cancelInteractive
Definition: actions.h:72
static TOOL_ACTION unselectAll
Definition: actions.h:83
static TOOL_ACTION selectItem
Select an item (specified as the event parameter).
Definition: actions.h:223
static TOOL_ACTION cursorLeft
Definition: actions.h:172
static TOOL_ACTION zoomOutCenter
Definition: actions.h:135
static TOOL_ACTION unselectItem
Definition: actions.h:224
static TOOL_ACTION zoomIn
Definition: actions.h:132
static TOOL_ACTION cursorLeftFast
Definition: actions.h:177
static TOOL_ACTION selectionCursor
Select a single item under the cursor position.
Definition: actions.h:217
static TOOL_ACTION groupEnter
Definition: actions.h:238
static TOOL_ACTION selectColumns
Definition: actions.h:102
static TOOL_ACTION cursorDown
Definition: actions.h:171
static TOOL_ACTION zoomOut
Definition: actions.h:133
static TOOL_ACTION cursorRightFast
Definition: actions.h:178
static TOOL_ACTION zoomCenter
Definition: actions.h:140
static TOOL_ACTION panDown
Definition: actions.h:185
static TOOL_ACTION selectionActivate
Activation of the selection tool.
Definition: actions.h:214
static TOOL_ACTION cursorDownFast
Definition: actions.h:176
static TOOL_ACTION selectionMenu
Run a selection menu to select from a list of items.
Definition: actions.h:232
static TOOL_ACTION reselectItem
Definition: actions.h:225
static TOOL_ACTION selectRows
Definition: actions.h:101
static TOOL_ACTION cursorUpFast
Definition: actions.h:175
static TOOL_ACTION panLeft
Definition: actions.h:186
static TOOL_ACTION updateMenu
Definition: actions.h:264
static TOOL_ACTION doDelete
Definition: actions.h:85
static TOOL_ACTION zoomFitScreen
Definition: actions.h:141
static TOOL_ACTION increment
Definition: actions.h:94
static TOOL_ACTION selectionClear
Clear the current selection.
Definition: actions.h:220
static TOOL_ACTION panUp
Definition: actions.h:184
static TOOL_ACTION zoomFitObjects
Definition: actions.h:142
static TOOL_ACTION zoomInCenter
Definition: actions.h:134
static TOOL_ACTION panRight
Definition: actions.h:187
static TOOL_ACTION selectTable
Definition: actions.h:103
static TOOL_ACTION cursorUp
Cursor control with keyboard.
Definition: actions.h:170
static TOOL_ACTION groupLeave
Definition: actions.h:239
static TOOL_ACTION cursorRight
Definition: actions.h:173
static TOOL_ACTION selectAll
Definition: actions.h:82
static TOOL_ACTION unselectItems
Definition: actions.h:229
static TOOL_ACTION selectItems
Select a list of items (specified as the event parameter)
Definition: actions.h:228
Define the structure of a menu based on ACTIONs.
Definition: action_menu.h:49
void SetTitle(const wxString &aTitle) override
Set title for the menu.
Definition: action_menu.cpp:92
wxMenuItem * Add(const wxString &aLabel, int aId, BITMAPS aIcon)
Add a wxWidgets-style entry to the menu.
static const ADVANCED_CFG & GetCfg()
Get the singleton instance's config, which is shared by all consumers.
CROSS_PROBING_SETTINGS m_CrossProbing
Definition: app_settings.h:193
BASE_SET & reset(size_t pos)
Definition: base_set.h:143
BASE_SET & set(size_t pos)
Definition: base_set.h:116
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
static bool ClassOf(const EDA_ITEM *aItem)
Returns information if the object is derived from BOARD_CONNECTED_ITEM.
Tool for pcb inspection.
int ClearHighlight(const TOOL_EVENT &aEvent)
Perform the appropriate action in response to an Eeschema cross-probe.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
virtual PCB_LAYER_ID GetLayer() const
Return the primary layer this item is on.
Definition: board_item.h:229
virtual bool IsConnected() const
Returns information if the object is derived from BOARD_CONNECTED_ITEM.
Definition: board_item.h:131
virtual VECTOR2I GetCenter() const
This defaults to the center of the bounding box if not overridden.
Definition: board_item.h:109
virtual std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer=UNDEFINED_LAYER, FLASHING aFlash=FLASHING::DEFAULT) const
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
Definition: board_item.cpp:279
FOOTPRINT * GetParentFootprint() const
Definition: board_item.cpp:299
virtual LSET GetLayerSet() const
Return a std::bitset of all layers on which the item physically resides.
Definition: board_item.h:249
virtual bool IsLocked() const
Definition: board_item.cpp:76
virtual void RunOnChildren(const std::function< void(BOARD_ITEM *)> &aFunction, RECURSE_MODE aMode) const
Invoke a function on all children.
Definition: board_item.h:205
BOARD_ITEM_CONTAINER * GetParent() const
Definition: board_item.h:207
virtual bool IsOnCopperLayer() const
Definition: board_item.h:148
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
INSPECT_RESULT Visit(INSPECTOR inspector, void *testData, const std::vector< KICAD_T > &scanTypes) override
May be re-implemented for each derived class in order to handle all the types given by its member dat...
Definition: board.cpp:1862
ZONE * m_SolderMaskBridges
Definition: board.h:1351
LSET GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
Definition: board.cpp:829
LSET GetVisibleLayers() const
A proxy function that calls the correspondent function in m_BoardSettings.
Definition: board.cpp:843
bool IsElementVisible(GAL_LAYER_ID aLayer) const
Test whether a given element category is visible.
Definition: board.cpp:895
bool IsLayerVisible(PCB_LAYER_ID aLayer) const
A proxy function that calls the correspondent function in m_BoardSettings tests whether a given layer...
Definition: board.cpp:835
bool LegacyTeardrops() const
Definition: board.h:1292
std::shared_ptr< CONNECTIVITY_DATA > GetConnectivity() const
Return a list of missing connections between components/tracks.
Definition: board.h:495
constexpr void SetMaximum()
Definition: box2.h:80
constexpr BOX2< Vec > & Inflate(coord_type dx, coord_type dy)
Inflates the rectangle horizontally by dx and vertically by dy.
Definition: box2.h:558
constexpr BOX2< Vec > & Normalize()
Ensure that the height and width are positive.
Definition: box2.h:146
constexpr size_type GetWidth() const
Definition: box2.h:214
constexpr Vec Centre() const
Definition: box2.h:97
constexpr size_type GetHeight() const
Definition: box2.h:215
constexpr bool Contains(const Vec &aPoint) const
Definition: box2.h:168
constexpr const Vec & GetOrigin() const
Definition: box2.h:210
constexpr const SizeVec & GetSize() const
Definition: box2.h:206
CN_ANCHOR represents a physical location that can be connected: a pad or a track/arc/via endpoint.
bool Dirty() const
BOARD_CONNECTED_ITEM * Parent() const
CN_EDGE represents a point-to-point connection, whether realized or unrealized (ie: tracks etc.
virtual double OnePixelInIU() const =0
void Transfer(int aIndex)
Move the item at aIndex (first position is 0) to the backup list.
Definition: collector.h:152
void Empty()
Clear the list.
Definition: collector.h:90
ITER begin()
Definition: collector.h:74
int GetCount() const
Return the number of objects in the list.
Definition: collector.h:82
bool HasItem(const EDA_ITEM *aItem) const
Tests if aItem has already been collected.
Definition: collector.h:196
void Remove(int aIndex)
Remove the item at aIndex (first position is 0).
Definition: collector.h:110
ITER end()
Definition: collector.h:75
void Append(EDA_ITEM *item)
Add an item to the end of the list.
Definition: collector.h:100
int ShowModal() override
bool IsType(FRAME_T aType) const
void AddStandardSubMenus(TOOL_MENU &aMenu)
Construct a "basic" menu for a tool, containing only items that apply to all tools (e....
void FocusOnLocation(const VECTOR2I &aPos)
Useful to focus on a particular location, in find functions.
virtual void ClearFocus()
void ForceRefresh()
Force a redraw.
void SetCurrentCursor(KICURSOR aCursor)
Set the current cursor shape for this panel.
A set of EDA_ITEMs (i.e., without duplicates).
Definition: eda_group.h:45
virtual EDA_ITEM * AsEdaItem()=0
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
virtual VECTOR2I GetPosition() const
Definition: eda_item.h:252
virtual const BOX2I GetBoundingBox() const
Return the orthogonal bounding box of this object for display purposes.
Definition: eda_item.cpp:84
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition: eda_item.h:135
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
void ClearSelected()
Definition: eda_item.h:130
void ClearFlags(EDA_ITEM_FLAGS aMask=EDA_ITEM_ALL_FLAGS)
Definition: eda_item.h:137
EDA_GROUP * GetParentGroup() const
Definition: eda_item.h:114
bool IsSelected() const
Definition: eda_item.h:120
void SetSelected()
Definition: eda_item.h:127
virtual bool IsType(const std::vector< KICAD_T > &aScanTypes) const
Check whether the item is one of the listed types.
Definition: eda_item.h:184
void ClearBrightened()
Definition: eda_item.h:131
void SetBrightened()
Definition: eda_item.h:128
virtual bool HitTest(const VECTOR2I &aPosition, int aAccuracy=0) const
Test if aPosition is inside or on the boundary of this item.
Definition: eda_item.h:225
EDA_ITEM * GetParent() const
Definition: eda_item.h:110
bool HasFlag(EDA_ITEM_FLAGS aFlag) const
Definition: eda_item.h:139
EDA_ITEM_FLAGS GetFlags() const
Definition: eda_item.h:138
bool IsMoving() const
Definition: eda_item.h:118
SHAPE_T GetShape() const
Definition: eda_shape.h:168
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
Definition: eda_shape.h:215
bool IsClosed() const
Definition: eda_shape.cpp:505
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
Definition: eda_shape.h:173
virtual bool IsVisible() const
Definition: eda_text.h:174
std::shared_ptr< SHAPE_COMPOUND > GetEffectiveTextShape(bool aTriangulate=true, const BOX2I &aBBox=BOX2I(), const EDA_ANGLE &aAngle=ANGLE_0) const
build a list of segments (SHAPE_SEGMENT) to describe a text shape.
Definition: eda_text.cpp:1120
static const TOOL_EVENT DisambiguatePoint
Used for hotkey feedback.
Definition: actions.h:353
static const TOOL_EVENT ClearedEvent
Definition: actions.h:338
static const TOOL_EVENT InhibitSelectionEditing
Definition: actions.h:349
static const TOOL_EVENT SelectedEvent
Definition: actions.h:336
static const TOOL_EVENT SelectedItemsModified
Selected items were moved, this can be very high frequency on the canvas, use with care.
Definition: actions.h:343
static const TOOL_EVENT UninhibitSelectionEditing
Used to inform tool that it should display the disambiguation menu.
Definition: actions.h:350
static const TOOL_EVENT PointSelectedEvent
Definition: actions.h:335
static const TOOL_EVENT SelectedItemsMoved
Used to inform tools that the selection should temporarily be non-editable.
Definition: actions.h:346
static const TOOL_EVENT UnselectedEvent
Definition: actions.h:337
ZONES & Zones()
Definition: footprint.h:217
static double GetCoverageArea(const BOARD_ITEM *aItem, const GENERAL_COLLECTOR &aCollector)
Definition: footprint.cpp:2779
PCB_LAYER_ID GetSide() const
Use instead of IsFlipped() when you also need to account for unsided footprints (those purely on user...
Definition: footprint.cpp:1761
const BOX2I GetLayerBoundingBox(const LSET &aLayers) const
Return the bounding box of the footprint on a given set of layers.
Definition: footprint.cpp:1486
std::deque< PAD * > & Pads()
Definition: footprint.h:211
PCB_LAYER_ID GetLayer() const override
Return the primary layer this item is on.
Definition: footprint.h:241
bool IsFlipped() const
Definition: footprint.h:396
SHAPE_POLY_SET GetBoundingHull() const
Return a bounding polygon for the shapes and pads in the footprint.
Definition: footprint.cpp:1531
bool IsLocked() const override
Definition: footprint.h:416
bool HitTestOnLayer(const VECTOR2I &aPosition, PCB_LAYER_ID aLayer, int aAccuracy=0) const
Test if the point hits one or more of the footprint elements on a given layer.
Definition: footprint.cpp:1823
const KIID_PATH & GetPath() const
Definition: footprint.h:268
DRAWINGS & GraphicalItems()
Definition: footprint.h:214
const BOX2I GetBoundingBox() const override
Return the orthogonal bounding box of this object for display purposes.
Definition: footprint.cpp:1336
A general implementation of a COLLECTORS_GUIDE.
Definition: collectors.h:319
void SetIgnoreBlindBuriedVias(bool ignore)
Definition: collectors.h:458
void SetIgnoreTracks(bool ignore)
Definition: collectors.h:464
void SetIgnoreFootprintsOnFront(bool ignore)
Definition: collectors.h:422
void SetIgnoreFPTextOnFront(bool ignore)
Definition: collectors.h:410
void SetIgnoreMicroVias(bool ignore)
Definition: collectors.h:461
void SetIgnoreZoneFills(bool ignore)
Definition: collectors.h:467
void SetIgnorePadsOnBack(bool ignore)
Definition: collectors.h:428
void SetIgnoreFPTextOnBack(bool ignore)
Definition: collectors.h:404
void SetLayerVisibleBits(const LSET &aLayerBits)
Definition: collectors.h:378
void SetIgnoreThroughVias(bool ignore)
Definition: collectors.h:455
void SetIgnoreThroughHolePads(bool ignore)
Definition: collectors.h:440
void SetIgnoreFPReferences(bool ignore)
Definition: collectors.h:452
void SetIgnoreFPValues(bool ignore)
Definition: collectors.h:446
void SetIgnorePadsOnFront(bool ignore)
Definition: collectors.h:434
void SetIgnoreFootprintsOnBack(bool ignore)
Definition: collectors.h:416
Used when the right click button is pressed, or when the select tool is in effect.
Definition: collectors.h:202
void SetGuide(const COLLECTORS_GUIDE *aGuide)
Record which COLLECTORS_GUIDE to use.
Definition: collectors.h:286
const COLLECTORS_GUIDE * GetGuide() const
Definition: collectors.h:288
static const std::vector< KICAD_T > AllBoardItems
A scan list for all editable board items.
Definition: collectors.h:222
void Collect(BOARD_ITEM *aItem, const std::vector< KICAD_T > &aScanList, const VECTOR2I &aRefPos, const COLLECTORS_GUIDE &aGuide)
Scan a BOARD_ITEM using this class's Inspector method, which does the collection.
Definition: collectors.cpp:482
static const std::vector< KICAD_T > FootprintItems
A scan list for primary footprint items.
Definition: collectors.h:248
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
virtual RENDER_SETTINGS * GetSettings()=0
Return a pointer to current settings that are going to be used when drawing items.
virtual void Update(const VIEW_ITEM *aItem, int aUpdateFlags) const override
For dynamic VIEWs, inform the associated VIEW that the graphical representation of this item has chan...
Definition: pcb_view.cpp:91
virtual void Add(VIEW_ITEM *aItem, int aDrawPriority=-1) override
Add a VIEW_ITEM to the view.
Definition: pcb_view.cpp:57
virtual void Remove(VIEW_ITEM *aItem) override
Remove a VIEW_ITEM from the view.
Definition: pcb_view.cpp:74
Represent a selection area (currently a rectangle) in a VIEW, drawn corner-to-corner between two poin...
void SetSubtractive(bool aSubtractive)
void SetAdditive(bool aAdditive)
void SetOrigin(const VECTOR2I &aOrigin)
const BOX2I ViewBBox() const override
Set the origin of the rectangle (the fixed corner)
void SetExclusiveOr(bool aExclusiveOr)
void SetEnd(const VECTOR2I &aEnd)
Set the current end of the rectangle (the corner that moves with the cursor.
Container for all the knowledge about how graphical objects are drawn on any output surface/device.
const std::set< int > & GetHighlightNetCodes() const
Return the netcode of currently highlighted net.
const std::set< int > GetHighContrastLayers() const
Returns the set of currently high-contrast layers.
virtual COLOR4D GetColor(const VIEW_ITEM *aItem, int aLayer) const =0
Returns the color that should be used to draw the specific VIEW_ITEM on the specific layer using curr...
bool GetHighContrast() const
void SetHighlight(bool aEnabled, int aNetcode=-1, bool aMulti=false)
Turns on/off highlighting.
virtual void SetAutoPan(bool aEnabled)
Turn on/off auto panning (this feature is used when there is a tool active (eg.
virtual void Clear()
Remove all the stored items from the group.
Definition: view_group.cpp:70
virtual void Add(VIEW_ITEM *aItem)
Add an item to the group.
Definition: view_group.cpp:58
An abstract base class for deriving all objects that can be added to a VIEW.
Definition: view_item.h:86
bool IsBOARD_ITEM() const
Definition: view_item.h:102
Hold a (potentially large) number of VIEW_ITEMs and renders them on a graphics device provided by the...
Definition: view.h:67
double GetScale() const
Definition: view.h:272
BOX2D GetViewport() const
Return the current viewport visible area rectangle.
Definition: view.cpp:520
virtual void SetScale(double aScale, VECTOR2D aAnchor={ 0, 0 })
Set the scaling factor, zooming around a given anchor point.
Definition: view.cpp:560
virtual void Remove(VIEW_ITEM *aItem)
Remove a VIEW_ITEM from the view.
Definition: view.cpp:332
void UpdateAllLayersColor()
Apply the new coloring scheme to all layers.
Definition: view.cpp:765
int Query(const BOX2I &aRect, std::vector< LAYER_ITEM_PAIR > &aResult) const
Find all visible items that touch or are within the rectangle aRect.
Definition: view.cpp:410
virtual void Update(const VIEW_ITEM *aItem, int aUpdateFlags) const
For dynamic VIEWs, inform the associated VIEW that the graphical representation of this item has chan...
Definition: view.cpp:1673
VECTOR2D ToWorld(const VECTOR2D &aCoord, bool aAbsolute=true) const
Converts a screen space point/vector to a point/vector in world space coordinates.
Definition: view.cpp:457
bool IsMirroredX() const
Return true if view is flipped across the X axis.
Definition: view.h:246
std::pair< VIEW_ITEM *, int > LAYER_ITEM_PAIR
Definition: view.h:71
void Hide(VIEW_ITEM *aItem, bool aHide=true, bool aHideOverlay=false)
Temporarily hide the item in the view (e.g.
Definition: view.cpp:1621
bool IsLayerVisible(int aLayer) const
Return information about visibility of a particular layer.
Definition: view.h:418
PAINTER * GetPainter() const
Return the painter object used by the view for drawing #VIEW_ITEMS.
Definition: view.h:216
void SetCenter(const VECTOR2D &aCenter)
Set the center point of the VIEW (i.e.
Definition: view.cpp:586
void MarkTargetDirty(int aTarget)
Set or clear target 'dirty' flag.
Definition: view.h:635
bool IsVisible(const VIEW_ITEM *aItem) const
Return information if the item is visible (or not).
Definition: view.cpp:1643
void SetVisible(VIEW_ITEM *aItem, bool aIsVisible=true)
Set the item visibility.
Definition: view.cpp:1600
wxString AsString() const
Definition: kiid.cpp:356
LSEQ is a sequence (and therefore also a set) of PCB_LAYER_IDs.
Definition: lseq.h:47
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
static LSET AllLayersMask()
Definition: lset.cpp:601
static LSET AllCuMask(int aCuLayerCount=MAX_CU_LAYERS)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Definition: lset.cpp:572
static LSET PhysicalLayersMask()
Return a mask holding all layers which are physically realized.
Definition: lset.cpp:658
LSEQ Seq(const LSEQ &aSequence) const
Return an LSEQ from the union of this LSET and a desired sequence.
Definition: lset.cpp:297
LSEQ SeqStackupTop2Bottom(PCB_LAYER_ID aSelectedLayer=UNDEFINED_LAYER) const
Generate a sequence of layers that represent a top to bottom stack of this set of layers.
Definition: lset.cpp:327
static LSET FrontMask()
Return a mask holding all technical layers and the external CU layer on front side.
Definition: lset.cpp:683
static LSET BackMask()
Return a mask holding all technical layers and the external CU layer on back side.
Definition: lset.cpp:690
bool Contains(PCB_LAYER_ID aLayer) const
See if the layer set contains a PCB layer.
Definition: lset.h:63
const VECTOR2I & GetPos() const
Definition: marker_base.h:83
void ShapeToPolygon(SHAPE_LINE_CHAIN &aPolygon, int aScale=-1) const
Return the shape polygon in internal units in a SHAPE_LINE_CHAIN the coordinates are relatives to the...
Handle the data for a net.
Definition: netinfo.h:56
Definition: pad.h:54
TRACK_DRAG_ACTION m_TrackDragAction
LOCKING_OPTIONS m_LockingOptions
static TOOL_ACTION drag45Degree
Definition: pcb_actions.h:180
static TOOL_ACTION unrouteSelected
Removes all tracks from the selected items to the first pad.
Definition: pcb_actions.h:76
static TOOL_ACTION grabUnconnected
Select and move nearest unconnected footprint from ratsnest of selection.
Definition: pcb_actions.h:88
static TOOL_ACTION filterSelection
Filter the items in the current selection (invokes dialog)
Definition: pcb_actions.h:100
static TOOL_ACTION highlightNet
Definition: pcb_actions.h:547
static TOOL_ACTION hideLocalRatsnest
Definition: pcb_actions.h:557
static TOOL_ACTION properties
Activation of the edit tool.
Definition: pcb_actions.h:163
static TOOL_ACTION selectOnSheetFromEeschema
Select all components on sheet from Eeschema crossprobing.
Definition: pcb_actions.h:91
static TOOL_ACTION selectConnection
Select tracks between junctions or expands an existing selection to pads or the entire connection.
Definition: pcb_actions.h:73
static TOOL_ACTION dragFreeAngle
Definition: pcb_actions.h:181
static TOOL_ACTION clearHighlight
Definition: pcb_actions.h:546
static TOOL_ACTION selectUnconnected
Select unconnected footprints from ratsnest of selection.
Definition: pcb_actions.h:85
static TOOL_ACTION moveIndividually
move items one-by-one
Definition: pcb_actions.h:106
static TOOL_ACTION syncSelection
Sets selection to specified items, zooms to fit, if enabled.
Definition: pcb_actions.h:63
static TOOL_ACTION selectSameSheet
Select all components on the same sheet as the selected footprint.
Definition: pcb_actions.h:94
static TOOL_ACTION selectNet
Select all connections belonging to a single net.
Definition: pcb_actions.h:79
static TOOL_ACTION move
move or drag an item
Definition: pcb_actions.h:103
static TOOL_ACTION syncSelectionWithNets
Sets selection to specified items with connected nets, zooms to fit, if enabled.
Definition: pcb_actions.h:66
static TOOL_ACTION deselectNet
Remove all connections belonging to a single net from the active selection.
Definition: pcb_actions.h:82
static TOOL_ACTION selectOnSchematic
Select symbols/pins on schematic corresponding to selected footprints/pads.
Definition: pcb_actions.h:97
Base PCB main window class for Pcbnew, Gerbview, and CvPcb footprint viewer.
const PCB_DISPLAY_OPTIONS & GetDisplayOptions() const
Display options control the way tracks, vias, outlines and other things are shown (for instance solid...
PCBNEW_SETTINGS * GetPcbNewSettings() const
virtual PCB_LAYER_ID GetActiveLayer() const
PCB_DRAW_PANEL_GAL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
BOARD * GetBoard() const
double m_TrackOpacity
Opacity override for all tracks.
double m_FilledShapeOpacity
Opacity override for graphic shapes.
double m_ZoneOpacity
Opacity override for filled zone areas.
double m_ImageOpacity
Opacity override for user images.
double m_PadOpacity
Opacity override for SMD pads and PTHs.
double m_ViaOpacity
Opacity override for all types of via.
ZONE_DISPLAY_MODE m_ZoneDisplayMode
virtual KIGFX::PCB_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
bool IsReference() const
Definition: pcb_field.h:68
bool IsValue() const
Definition: pcb_field.h:69
A set of BOARD_ITEMs (i.e., without duplicates).
Definition: pcb_group.h:53
const BOX2I GetBoundingBox() const override
Return the orthogonal bounding box of this object for display purposes.
Definition: pcb_group.cpp:286
static bool WithinScope(BOARD_ITEM *aItem, PCB_GROUP *aScope, bool isFootprintEditor)
Definition: pcb_group.cpp:179
void RunOnChildren(const std::function< void(BOARD_ITEM *)> &aFunction, RECURSE_MODE aMode) const override
Invoke a function on all children.
Definition: pcb_group.cpp:423
static EDA_GROUP * TopLevelGroup(BOARD_ITEM *aItem, EDA_GROUP *aScope, bool isFootprintEditor)
Definition: pcb_group.cpp:173
std::unordered_set< BOARD_ITEM * > GetBoardItems() const
Definition: pcb_group.cpp:122
Tool that displays edit points allowing to modify items by dragging the points.
bool HasPoint()
Indicate the cursor is over an edit point.
Private implementation of firewalled private data.
DIALOG_FILTER_SELECTION::OPTIONS m_filterOpts
The selection tool: currently supports:
void highlight(EDA_ITEM *aItem, int aHighlightMode, SELECTION *aGroup=nullptr) override
Highlight the item visually.
int syncSelectionWithNets(const TOOL_EVENT &aEvent)
int SelectTable(const TOOL_EVENT &aEvent)
Clear current selection event handler.
PCB_BASE_FRAME * frame() const
int syncSelection(const TOOL_EVENT &aEvent)
int selectNet(const TOOL_EVENT &aEvent)
Select all copper connections belonging to the same net(s) as the items in the selection.
int filterSelection(const TOOL_EVENT &aEvent)
Return true if the given item passes the current SELECTION_FILTER_OPTIONS.
void Reset(RESET_REASON aReason) override
Bring the tool to a known, initial state.
void ZoomFitCrossProbeBBox(const BOX2I &bbox)
void doSyncSelection(const std::vector< BOARD_ITEM * > &aItems, bool aWithNets)
Invoke filter dialog and modify current selection.
void GuessSelectionCandidates(GENERAL_COLLECTOR &aCollector, const VECTOR2I &aWhere) const
Try to guess best selection candidates in case multiple items are clicked, by doing some brain-dead h...
bool itemPassesFilter(BOARD_ITEM *aItem, bool aMultiSelect)
bool isExpandableGraphicShape(const EDA_ITEM *aItem) const
int disambiguateCursor(const TOOL_EVENT &aEvent)
Handle disambiguation actions including displaying the menu.
void FilterCollectorForMarkers(GENERAL_COLLECTOR &aCollector) const
Drop any PCB_MARKERs from the collector.
int UnselectAll(const TOOL_EVENT &aEvent)
void unhighlightInternal(EDA_ITEM *aItem, int aHighlightMode, bool aUsingOverlay)
bool selectionContains(const VECTOR2I &aPoint) const
void select(EDA_ITEM *aItem) override
Take necessary action mark an item as selected.
bool selectCursor(bool aForceSelect=false, CLIENT_SELECTION_FILTER aClientFilter=nullptr)
Select an item under the cursor unless there is something already selected.
int SelectColumns(const TOOL_EVENT &aEvent)
std::unique_ptr< PRIV > m_priv
PCB_SELECTION & RequestSelection(CLIENT_SELECTION_FILTER aClientFilter, bool aConfirmLockedItems=false)
Return the current selection, filtered according to aClientFilter.
int unrouteSelected(const TOOL_EVENT &aEvent)
Unroute the selected board connected items.
SELECTION & selection() override
Return a reference to the selection.
int grabUnconnected(const TOOL_EVENT &aEvent)
Select and move other nearest footprint unconnected on same net as selected items.
void FilterCollectorForFreePads(GENERAL_COLLECTOR &aCollector, bool aForcePromotion=false) const
Check the "allow free pads" setting and if disabled, replace any pads in the collector with their par...
const GENERAL_COLLECTORS_GUIDE getCollectorsGuide() const
bool Selectable(const BOARD_ITEM *aItem, bool checkVisibilityOnly=false) const
void FilterCollectedItems(GENERAL_COLLECTOR &aCollector, bool aMultiSelect)
Apply the SELECTION_FITLER_OPTIONS to the collector.
bool selectPoint(const VECTOR2I &aWhere, bool aOnDrag=false, bool *aSelectionCancelledFlag=nullptr, CLIENT_SELECTION_FILTER aClientFilter=nullptr)
Select an item pointed by the parameter aWhere.
KIGFX::PCB_VIEW * view() const
void selectAllItemsOnSheet(wxString &aSheetPath)
Select all items with the given sheet timestamp/UUID name (the sheet path).
void FilterCollectorForHierarchy(GENERAL_COLLECTOR &aCollector, bool aMultiselect) const
In general we don't want to select both a parent and any of it's children.
void setTransitions() override
Zoom the screen to center and fit the current selection.
int expandConnection(const TOOL_EVENT &aEvent)
Expand the current connected-item selection to the next boundary (junctions, pads,...
int selectUnconnected(const TOOL_EVENT &aEvent)
Select nearest unconnected footprints on same net as selected items.
virtual bool ctrlClickHighlights() override
Determine if ctrl-click is highlight net or XOR selection.
int selectSheetContents(const TOOL_EVENT &aEvent)
Select all footprints belonging to same hierarchical sheet as the selected footprint (same sheet path...
int SelectRows(const TOOL_EVENT &aEvent)
int selectSameSheet(const TOOL_EVENT &aEvent)
Set selection to items passed by parameter and connected nets (optionally).
void EnterGroup()
Enter the group at the head of the current selection.
void zoomFitSelection()
Zoom the screen to fit the bounding box for cross probing/selection sync.
void selectAllConnectedTracks(const std::vector< BOARD_CONNECTED_ITEM * > &aStartItems, STOP_CONDITION aStopCondition)
Select connected tracks and vias.
int CursorSelection(const TOOL_EVENT &aEvent)
int ClearSelection(const TOOL_EVENT &aEvent)
void highlightInternal(EDA_ITEM *aItem, int aHighlightMode, bool aUsingOverlay)
PCB_BASE_FRAME * m_frame
PCB_SELECTION_FILTER_OPTIONS m_filter
PCB_SELECTION & GetSelection()
@ STOP_AT_PAD
Stop when reaching a pad.
@ STOP_NEVER
Select the entire net.
@ STOP_AT_JUNCTION
Stop at any place where more than two traces meet.
int Main(const TOOL_EVENT &aEvent)
The main loop.
void RebuildSelection()
Rebuild the selection from the EDA_ITEMs' selection flags.
void pruneObscuredSelectionCandidates(GENERAL_COLLECTOR &aCollector) const
void OnIdle(wxIdleEvent &aEvent)
PCB_DRAW_PANEL_GAL * canvas() const
void FilterCollectorForFootprints(GENERAL_COLLECTOR &aCollector, const VECTOR2I &aWhere) const
Drop footprints that are not directly selected.
void FindItem(BOARD_ITEM *aItem)
Take necessary actions to mark an item as found.
int updateSelection(const TOOL_EVENT &aEvent)
Event handler to update the selection VIEW_ITEM.
bool Init() override
Init() is called once upon a registration of the tool.
BOARD * board() const
KIGFX::VIEW_GROUP m_enteredGroupOverlay
void selectConnections(const std::vector< BOARD_ITEM * > &aItems)
int hitTestDistance(const VECTOR2I &aWhere, BOARD_ITEM *aItem, int aMaxDistance) const
void selectAllConnectedShapes(const std::vector< PCB_SHAPE * > &aStartItems)
Select all non-closed shapes that are graphically connected to the given start items.
void SelectAllItemsOnNet(int aNetCode, bool aSelect=true)
Select all items with the given net code.
void FilterCollectorForTableCells(GENERAL_COLLECTOR &aCollector) const
Promote any table cell selections to the whole table.
bool selectMultiple()
Handle drawing a selection box that allows one to select many items at the same time.
void ExitGroup(bool aSelectGroup=false)
Leave the currently-entered group.
void unselect(EDA_ITEM *aItem) override
Take necessary action mark an item as unselected.
int SelectAll(const TOOL_EVENT &aEvent)
Unselect all items on the board.
PCB_SELECTION m_selection
bool selectTableCells(PCB_TABLE *aTable)
void unhighlight(EDA_ITEM *aItem, int aHighlightMode, SELECTION *aGroup=nullptr) override
Unhighlight the item visually.
BOX2I GetBoundingBox() const override
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition: pcb_shape.cpp:227
std::vector< VECTOR2I > GetConnectionPoints() const
Definition: pcb_shape.cpp:257
int GetRowSpan() const
Definition: pcb_tablecell.h:65
int GetColSpan() const
Definition: pcb_tablecell.h:62
std::vector< PCB_TABLECELL * > GetCells() const
Definition: pcb_table.h:150
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition: pcb_track.cpp:1206
const VECTOR2I & GetStart() const
Definition: pcb_track.h:152
const VECTOR2I & GetEnd() const
Definition: pcb_track.h:149
int GetDrill() const
Return the local drill setting for this PCB_VIA.
Definition: pcb_track.h:668
A small class to help profiling.
Definition: profile.h:49
void Start()
Start or restart the counter.
Definition: profile.h:77
double msecs(bool aSinceLast=false)
Definition: profile.h:149
bool RoutingInProgress()
Returns whether routing is currently active.
Definition: seg.h:42
static SEG::ecoord Square(int a)
Definition: seg.h:123
static SELECTION_CONDITION HasType(KICAD_T aType)
Create a functor that tests if among the selected items there is at least one of a given type.
static bool NotEmpty(const SELECTION &aSelection)
Test if there are any items selected.
static SELECTION_CONDITION MoreThan(int aNumber)
Create a functor that tests if the number of selected items is greater than the value given as parame...
static SELECTION_CONDITION Count(int aNumber)
Create a functor that tests if the number of selected items is equal to the value given as parameter.
static SELECTION_CONDITION OnlyTypes(std::vector< KICAD_T > aTypes)
Create a functor that tests if the selected items are only of given types.
bool m_multiple
Multiple selection mode is active.
int RemoveItemFromSel(const TOOL_EVENT &aEvent)
bool doSelectionMenu(COLLECTOR *aCollector)
wxTimer m_disambiguateTimer
Timer to show the disambiguate menu.
bool m_drag_additive
Add multiple items to selection.
bool m_exclusive_or
Items' selection state should be toggled.
int AddItemsToSel(const TOOL_EVENT &aEvent)
int AddItemToSel(const TOOL_EVENT &aEvent)
int UpdateMenu(const TOOL_EVENT &aEvent)
Update a menu's state based on the current selection.
void setModifiersState(bool aShiftState, bool aCtrlState, bool aAltState)
Set the configuration of m_additive, m_subtractive, m_exclusive_or, m_skip_heuristics from the state ...
VECTOR2I m_originalCursor
Location of original cursor when starting click.
int SelectionMenu(const TOOL_EVENT &aEvent)
Show a popup menu to trim the COLLECTOR passed as aEvent's parameter down to a single item.
int RemoveItemsFromSel(const TOOL_EVENT &aEvent)
bool m_subtractive
Items should be removed from selection.
bool m_highlight_modifier
Select highlight net on left click.
bool m_skip_heuristics
Show disambiguation menu for all items under the cursor rather than trying to narrow them down first ...
int ReselectItem(const TOOL_EVENT &aEvent)
bool m_drag_subtractive
Remove multiple from selection.
bool m_additive
Items should be added to sel (instead of replacing).
bool hasModifier()
True if a selection modifier is enabled, false otherwise.
bool m_canceledMenu
Sets to true if the disambiguation menu was canceled.
void onDisambiguationExpire(wxTimerEvent &aEvent)
Start the process to show our disambiguation menu once the user has kept the mouse down for the minim...
virtual void Add(EDA_ITEM *aItem)
Definition: selection.cpp:42
virtual KIGFX::VIEW_ITEM * GetItem(unsigned int aIdx) const override
Definition: selection.cpp:75
const std::deque< EDA_ITEM * > GetItems() const
Definition: selection.h:121
void SetIsHover(bool aIsHover)
Definition: selection.h:79
virtual void Remove(EDA_ITEM *aItem)
Definition: selection.cpp:60
virtual unsigned int GetSize() const override
Return the number of stored items.
Definition: selection.h:100
EDA_ITEM * Front() const
Definition: selection.h:172
virtual void Clear() override
Remove all the stored items from the group.
Definition: selection.h:93
int Size() const
Returns the number of selected parts.
Definition: selection.h:116
void ClearReferencePoint()
Definition: selection.cpp:184
bool Empty() const
Checks if there is anything selected.
Definition: selection.h:110
size_t CountType(KICAD_T aType) const
Definition: selection.cpp:155
bool Contains(EDA_ITEM *aItem) const
Definition: selection.cpp:84
ACTION_MENU * create() const override
Return an instance of this class. It has to be overridden in inheriting classes.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
void Move(const VECTOR2I &aVector) override
virtual bool Collide(const VECTOR2I &aP, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const override
Check if point aP lies closer to us than aClearance.
bool Collide(const SHAPE *aShape, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const override
Check if the boundary of shape (this) lies closer to the shape aShape than aClearance,...
MOUSE_DRAG_ACTION GetDragAction() const
Indicates whether a drag should draw a selection rectangle or drag selected (or unselected) objects.
Definition: tools_holder.h:144
bool ToolStackIsEmpty()
Definition: tools_holder.h:125
Represent a single user action.
Definition: tool_action.h:304
KIGFX::VIEW_CONTROLS * getViewControls() const
Return the instance of VIEW_CONTROLS object used in the application.
Definition: tool_base.cpp:44
TOOL_MANAGER * m_toolMgr
Definition: tool_base.h:220
KIGFX::VIEW * getView() const
Returns the instance of #VIEW object used in the application.
Definition: tool_base.cpp:38
bool IsToolActive() const
Definition: tool_base.cpp:32
RESET_REASON
Determine the reason of reset for a tool.
Definition: tool_base.h:78
@ REDRAW
Full drawing refresh.
Definition: tool_base.h:83
@ MODEL_RELOAD
Model changes (the sheet for a schematic)
Definition: tool_base.h:80
Generic, UI-independent tool event.
Definition: tool_event.h:168
bool IsAction(const TOOL_ACTION *aAction) const
Test if the event contains an action issued upon activation of the given TOOL_ACTION.
Definition: tool_event.cpp:82
T Parameter() const
Return a parameter assigned to the event.
Definition: tool_event.h:465
void Go(int(T::*aStateFunc)(const TOOL_EVENT &), const TOOL_EVENT_LIST &aConditions=TOOL_EVENT(TC_ANY, TA_ANY))
Define which state (aStateFunc) to go when a certain event arrives (aConditions).
std::unique_ptr< TOOL_MENU > m_menu
The functions below are not yet implemented - their interface may change.
TOOL_EVENT * Wait(const TOOL_EVENT_LIST &aEventList=TOOL_EVENT(TC_ANY, TA_ANY))
Suspend execution of the tool until an event specified in aEventList arrives.
bool ProcessEvent(const TOOL_EVENT &aEvent)
Propagate an event to tools that requested events of matching type(s).
bool RunAction(const std::string &aActionName, T aParam)
Run the specified action immediately, pausing the current action to run the new one.
Definition: tool_manager.h:150
VECTOR2D GetMousePosition() const
void VetoContextMenuMouseWarp()
Disable mouse warping after the current context menu is closed.
Definition: tool_manager.h:515
KIGFX::VIEW * GetView() const
Definition: tool_manager.h:395
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
bool GetIsRuleArea() const
Accessors to parameters used in Rule Area zones:
Definition: zone.h:752
bool HitTestForCorner(const VECTOR2I &refPos, int aAccuracy, SHAPE_POLY_SET::VERTEX_INDEX *aCornerHit=nullptr) const
Test if the given VECTOR2I is near a corner.
Definition: zone.cpp:740
bool HitTestForEdge(const VECTOR2I &refPos, int aAccuracy, SHAPE_POLY_SET::VERTEX_INDEX *aCornerHit=nullptr) const
Test if the given VECTOR2I is near a segment defined by 2 corners.
Definition: zone.cpp:747
bool IsTeardropArea() const
Definition: zone.h:727
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition: zone.h:136
bool GetDoNotAllowZoneFills() const
Definition: zone.h:768
MOUSE_DRAG_ACTION
#define EXCLUDE_ZONES
#define IGNORE_NETS
Function GetConnectedItems() Returns a list of items connected to a source item aItem.
KICURSOR
Definition: cursors.h:34
#define DEFAULT_TEXT_SIZE
Ratio of the font height to the baseline of the text above the wire.
static bool empty(const wxTextEntryBase *aCtrl)
#define _(s)
std::function< INSPECT_RESULT(EDA_ITEM *aItem, void *aTestData) > INSPECTOR_FUNC
Used to inspect and possibly collect the (search) results of iterating over a list or tree of KICAD_T...
Definition: eda_item.h:86
#define BRIGHTENED
item is drawn with a bright contour
#define IS_NEW
New item, just created.
#define SELECTED
Item was manually selected by the user.
#define ENTERED
indicates a group has been entered
#define SKIP_STRUCT
flag indicating that the structure should be ignored
#define CANDIDATE
flag indicating that the structure is connected
#define IS_MOVING
Item being moved.
@ FRAME_PCB_EDITOR
Definition: frame_type.h:42
@ FRAME_FOOTPRINT_VIEWER
Definition: frame_type.h:45
@ FRAME_FOOTPRINT_EDITOR
Definition: frame_type.h:43
int m_DisambiguationMenuDelay
The number of milliseconds to wait in a click before showing a disambiguation menu.
double m_PcbSelectionVisibilityRatio
Board object selection visibility limit.
@ LAYER_FOOTPRINTS_FR
Show footprints on front.
Definition: layer_ids.h:258
@ LAYER_DRAW_BITMAPS
Draw images.
Definition: layer_ids.h:283
@ LAYER_FP_REFERENCES
Show footprints references (when texts are visible).
Definition: layer_ids.h:265
@ LAYER_ZONES
Control for copper zone opacity/visibility (color ignored).
Definition: layer_ids.h:294
@ LAYER_PADS
Meta control for all pads opacity/visibility (color ignored).
Definition: layer_ids.h:291
@ LAYER_TRACKS
Definition: layer_ids.h:266
@ LAYER_FP_TEXT
Definition: layer_ids.h:239
@ LAYER_FOOTPRINTS_BK
Show footprints on back.
Definition: layer_ids.h:259
@ LAYER_FP_VALUES
Show footprints values (when texts are visible).
Definition: layer_ids.h:262
@ LAYER_VIAS
Meta control for all vias opacity/visibility.
Definition: layer_ids.h:232
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ F_CrtYd
Definition: layer_ids.h:116
@ Edge_Cuts
Definition: layer_ids.h:112
@ F_SilkS
Definition: layer_ids.h:100
@ B_CrtYd
Definition: layer_ids.h:115
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
@ B_SilkS
Definition: layer_ids.h:101
@ PCB_LAYER_ID_COUNT
Definition: layer_ids.h:171
PCB_LAYER_ID ToLAYER_ID(int aLayer)
Definition: lset.cpp:723
This file contains miscellaneous commonly used macros and functions.
#define KI_FALLTHROUGH
The KI_FALLTHROUGH macro is to be used when switch statement cases should purposely fallthrough from ...
Definition: macros.h:83
@ REPAINT
Item needs to be redrawn.
Definition: view_item.h:58
@ TARGET_OVERLAY
Items that may change while the view stays the same (noncached)
Definition: definitions.h:39
STL namespace.
Class to handle a set of BOARD_ITEMs.
void connectedItemFilter(const VECTOR2I &, GENERAL_COLLECTOR &aCollector, PCB_SELECTION_TOOL *sTool)
const TOOL_ACTION * allowedActions[]
static bool itemIsIncludedByFilter(const BOARD_ITEM &aItem, const BOARD &aBoard, const DIALOG_FILTER_SELECTION::OPTIONS &aFilterOptions)
Determine if an item is included by the filter specified.
TRACK_DRAG_ACTION
CITER next(CITER it)
Definition: ptree.cpp:124
Class that computes missing connections on a PCB.
static float distance(const SFVEC2UI &a, const SFVEC2UI &b)
static std::vector< KICAD_T > tableCellTypes
const TOOL_ACTION * allowedActions[]
const int scale
bool zoom_to_fit
Zoom to fit items (ignored if center_on_items is off).
Definition: app_settings.h:35
bool center_on_items
Automatically pan to cross-probed items.
Definition: app_settings.h:34
Struct that will be set with the result of the user choices in the dialog.
constexpr int mmToIU(double mm) const
Definition: base_units.h:88
const BOARD_ITEM * m_Item
bool otherItems
Anything not fitting one of the above categories.
bool graphics
Graphic lines, shapes, polygons.
bool footprints
Allow selecting entire footprints.
bool text
Text (free or attached to a footprint)
bool lockedItems
Allow selecting locked items.
const int accuracy
constexpr int delta
@ TA_MOUSE_UP
Definition: tool_event.h:69
@ TA_MOUSE_WHEEL
Definition: tool_event.h:73
@ TC_ANY
Definition: tool_event.h:60
@ MD_ALT
Definition: tool_event.h:145
@ MD_CTRL
Definition: tool_event.h:144
@ MD_SHIFT
Definition: tool_event.h:143
@ BUT_MIDDLE
Definition: tool_event.h:134
@ BUT_LEFT
Definition: tool_event.h:132
@ BUT_RIGHT
Definition: tool_event.h:133
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
Definition: typeinfo.h:78
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
Definition: typeinfo.h:88
@ PCB_DIM_ORTHOGONAL_T
class PCB_DIM_ORTHOGONAL, a linear dimension constrained to x/y
Definition: typeinfo.h:105
@ PCB_DIM_LEADER_T
class PCB_DIM_LEADER, a leader dimension (graphic item)
Definition: typeinfo.h:102
@ PCB_GENERATOR_T
class PCB_GENERATOR, generator on a layer
Definition: typeinfo.h:91
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
Definition: typeinfo.h:97
@ TYPE_NOT_INIT
Definition: typeinfo.h:81
@ PCB_DIM_CENTER_T
class PCB_DIM_CENTER, a center point marking (graphic item)
Definition: typeinfo.h:103
@ PCB_GROUP_T
class PCB_GROUP, a set of BOARD_ITEMs
Definition: typeinfo.h:110
@ PCB_TEXTBOX_T
class PCB_TEXTBOX, wrapped text on a layer
Definition: typeinfo.h:93
@ PCB_ZONE_T
class ZONE, a copper pour area
Definition: typeinfo.h:107
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
Definition: typeinfo.h:92
@ PCB_REFERENCE_IMAGE_T
class PCB_REFERENCE_IMAGE, bitmap on a layer
Definition: typeinfo.h:89
@ PCB_FIELD_T
class PCB_FIELD, text associated with a footprint property
Definition: typeinfo.h:90
@ NOT_USED
the 3d code uses this value
Definition: typeinfo.h:79
@ PCB_MARKER_T
class PCB_MARKER, a marker used to show something
Definition: typeinfo.h:99
@ PCB_TARGET_T
class PCB_TARGET, a target (graphic item)
Definition: typeinfo.h:106
@ PCB_TABLECELL_T
class PCB_TABLECELL, PCB_TEXTBOX for use in tables
Definition: typeinfo.h:95
@ PCB_FOOTPRINT_T
class FOOTPRINT, a footprint
Definition: typeinfo.h:86
@ PCB_DIM_ALIGNED_T
class PCB_DIM_ALIGNED, a linear dimension (graphic item)
Definition: typeinfo.h:101
@ PCB_PAD_T
class PAD, a pad in a footprint
Definition: typeinfo.h:87
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
Definition: typeinfo.h:98
@ PCB_TABLE_T
class PCB_TABLE, table of PCB_TABLECELLs
Definition: typeinfo.h:94
@ PCB_NETINFO_T
class NETINFO_ITEM, a description of a net
Definition: typeinfo.h:109
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
Definition: typeinfo.h:96
@ PCB_DIM_RADIAL_T
class PCB_DIM_RADIAL, a radius or diameter dimension
Definition: typeinfo.h:104
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695
VECTOR2D ToVECTOR2D(const wxPoint &aPoint)
Definition: vector2wx.h:40