KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcb_control.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) 2014-2016 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Maciej Suminski <maciej.suminski@cern.ch>
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26#include "pcb_control.h"
28
29#include <kiplatform/ui.h>
30#include <tools/edit_tool.h>
32#include <router/router_tool.h>
33#include <pgm_base.h>
34#include <tools/pcb_actions.h>
39#include <board_commit.h>
40#include <board.h>
42#include <board_item.h>
43#include <clipboard.h>
44#include <design_block.h>
46#include <pcb_dimension.h>
48#include <footprint.h>
49#include <layer_pairs.h>
50#include <pcb_group.h>
52#include <pcb_reference_image.h>
53#include <pcb_textbox.h>
54#include <pcb_track.h>
55#include <pcb_generator.h>
57#include <zone.h>
58#include <confirm.h>
59#include <kidialog.h>
61#include <core/kicad_algo.h>
63#include <kicad_clipboard.h>
64#include <origin_viewitem.h>
65#include <pcb_edit_frame.h>
66#include <pcb_painter.h>
68#include <string>
69#include <tool/tool_manager.h>
76#include <widgets/wx_infobar.h>
77#include <wx/hyperlink.h>
78
79
80using namespace std::placeholders;
81
82
83// files.cpp
84extern bool AskLoadBoardFileName( PCB_EDIT_FRAME* aParent, wxString* aFileName, int aCtl = 0 );
85
86
88 PCB_TOOL_BASE( "pcbnew.Control" ),
89 m_frame( nullptr ),
90 m_pickerItem( nullptr )
91{
93}
94
95
97{
98}
99
100
102{
103 m_frame = getEditFrame<PCB_BASE_FRAME>();
104
105 if( aReason == MODEL_RELOAD || aReason == GAL_SWITCH || aReason == REDRAW )
106 {
107 m_gridOrigin->SetPosition( board()->GetDesignSettings().GetGridOrigin() );
108 m_gridOrigin->SetColor( m_frame->GetGridColor() );
109 getView()->Remove( m_gridOrigin.get() );
110 getView()->Add( m_gridOrigin.get() );
111 }
112}
113
114
116{
118 {
119 if( aEvent.IsAction( &ACTIONS::newLibrary ) )
120 static_cast<PCB_BASE_EDIT_FRAME*>( m_frame )->CreateNewLibrary();
121 else if( aEvent.IsAction( &ACTIONS::addLibrary ) )
122 static_cast<PCB_BASE_EDIT_FRAME*>( m_frame )->AddLibrary();
123 }
124
125 return 0;
126}
127
128
130{
132 static_cast<FOOTPRINT_EDIT_FRAME*>( m_frame )->LoadFootprintFromBoard( nullptr );
133
134 return 0;
135}
136
137
139{
141 static_cast<FOOTPRINT_EDIT_FRAME*>( m_frame )->SaveFootprintToBoard( true );
144
145 return 0;
146}
147
148
150{
151 const wxString fn = *aEvent.Parameter<wxString*>();
152 static_cast<PCB_BASE_EDIT_FRAME*>( m_frame )->AddLibrary( fn );
153 return 0;
154}
155
156
158{
159 const wxString fn = *aEvent.Parameter<wxString*>();
160 static_cast<FOOTPRINT_EDIT_FRAME*>( m_frame )->ImportFootprint( fn );
161 m_frame->Zoom_Automatique( false );
162 return 0;
163}
164
165
167{
169 static_cast<FOOTPRINT_VIEWER_FRAME*>( m_frame )->SelectAndViewFootprint( aEvent.Parameter<FPVIEWER_CONSTANTS>() );
170
171 return 0;
172}
173
174
175int PCB_CONTROL::Quit( const TOOL_EVENT& aEvent )
176{
177 m_frame->Close( false );
178 return 0;
179}
180
181
182template<class T> void Flip( T& aValue )
183{
184 aValue = !aValue;
185}
186
187
189{
190 Flip( displayOptions().m_DisplayPcbTrackFill );
191
192 for( PCB_TRACK* track : board()->Tracks() )
193 {
194 if( track->Type() == PCB_TRACE_T || track->Type() == PCB_ARC_T )
195 view()->Update( track, KIGFX::REPAINT );
196 }
197
198 for( BOARD_ITEM* shape : board()->Drawings() )
199 {
200 if( shape->Type() == PCB_SHAPE_T && static_cast<PCB_SHAPE*>( shape )->IsOnCopperLayer() )
201 view()->Update( shape, KIGFX::REPAINT );
202 }
203
204 canvas()->Refresh();
205
206 return 0;
207}
208
209
211{
212 if( aEvent.IsAction( &PCB_ACTIONS::showRatsnest ) )
213 {
214 // N.B. Do not disable the Ratsnest layer here. We use it for local ratsnest
215 Flip( displayOptions().m_ShowGlobalRatsnest );
216 getEditFrame<PCB_EDIT_FRAME>()->SetElementVisibility( LAYER_RATSNEST,
217 displayOptions().m_ShowGlobalRatsnest );
218
219 }
220 else if( aEvent.IsAction( &PCB_ACTIONS::ratsnestLineMode ) )
221 {
222 Flip( displayOptions().m_DisplayRatsnestLinesCurved );
223 }
224
225 frame()->OnDisplayOptionsChanged();
226
228 canvas()->Refresh();
229
230 return 0;
231}
232
233
235{
236 Flip( displayOptions().m_DisplayViaFill );
237
238 for( PCB_TRACK* track : board()->Tracks() )
239 {
240 if( track->Type() == PCB_VIA_T )
241 view()->Update( track, KIGFX::REPAINT );
242 }
243
244 canvas()->Refresh();
245 return 0;
246}
247
248
255{
256 if( Pgm().GetCommonSettings()->m_DoNotShowAgain.zone_fill_warning )
257 return;
258
259 bool unfilledZones = false;
260
261 for( const ZONE* zone : board()->Zones() )
262 {
263 if( !zone->GetIsRuleArea() && !zone->IsFilled() )
264 {
265 unfilledZones = true;
266 break;
267 }
268 }
269
270 if( unfilledZones )
271 {
272 WX_INFOBAR* infobar = frame()->GetInfoBar();
273 wxHyperlinkCtrl* button = new wxHyperlinkCtrl( infobar, wxID_ANY, _( "Don't show again" ),
274 wxEmptyString );
275
276 button->Bind( wxEVT_COMMAND_HYPERLINK, std::function<void( wxHyperlinkEvent& aEvent )>(
277 [&]( wxHyperlinkEvent& aEvent )
278 {
280 frame()->GetInfoBar()->Dismiss();
281 } ) );
282
283 infobar->RemoveAllButtons();
284 infobar->AddButton( button );
285
286 wxString msg;
287 msg.Printf( _( "Not all zones are filled. Use Edit > Fill All Zones (%s) "
288 "if you wish to see all fills." ),
290
291 infobar->ShowMessageFor( msg, 5000, wxICON_WARNING );
292 }
293}
294
295
297{
298 PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions();
299
300 // Apply new display options to the GAL canvas
302 {
304
305 opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_FILLED;
306 }
307 else if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayOutline ) )
308 {
309 opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_ZONE_OUTLINE;
310 }
311 else if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayFractured ) )
312 {
313 opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_FRACTURE_BORDERS;
314 }
316 {
317 opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_TRIANGULATION;
318 }
319 else if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayToggle ) )
320 {
321 if( opts.m_ZoneDisplayMode == ZONE_DISPLAY_MODE::SHOW_FILLED )
322 opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_ZONE_OUTLINE;
323 else
324 opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_FILLED;
325 }
326 else
327 {
328 wxFAIL;
329 }
330
331 m_frame->SetDisplayOptions( opts );
332
333 for( ZONE* zone : board()->Zones() )
334 view()->Update( zone, KIGFX::REPAINT );
335
336 canvas()->Refresh();
337
338 return 0;
339}
340
341
343{
344 PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions();
345
346 opts.m_ContrastModeDisplay = opts.m_ContrastModeDisplay == HIGH_CONTRAST_MODE::NORMAL
347 ? HIGH_CONTRAST_MODE::DIMMED
348 : HIGH_CONTRAST_MODE::NORMAL;
349
350 m_frame->SetDisplayOptions( opts );
351 return 0;
352}
353
354
356{
357 PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions();
358
359 switch( opts.m_ContrastModeDisplay )
360 {
361 case HIGH_CONTRAST_MODE::NORMAL: opts.m_ContrastModeDisplay = HIGH_CONTRAST_MODE::DIMMED; break;
362 case HIGH_CONTRAST_MODE::DIMMED: opts.m_ContrastModeDisplay = HIGH_CONTRAST_MODE::HIDDEN; break;
363 case HIGH_CONTRAST_MODE::HIDDEN: opts.m_ContrastModeDisplay = HIGH_CONTRAST_MODE::NORMAL; break;
364 }
365
366 m_frame->SetDisplayOptions( opts );
367
369 return 0;
370}
371
372
374{
375 if( !Pgm().GetCommonSettings()->m_Input.hotkey_feedback )
376 return 0;
377
378 PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions();
379
380 wxArrayString labels;
381 labels.Add( _( "Normal" ) );
382 labels.Add( _( "Dimmed" ) );
383 labels.Add( _( "Hidden" ) );
384
385 if( !m_frame->GetHotkeyPopup() )
387
389
390 if( popup )
391 {
392 popup->Popup( _( "Inactive Layer Display" ), labels,
393 static_cast<int>( opts.m_ContrastModeDisplay ) );
394 }
395
396 return 0;
397}
398
399
401{
402 PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions();
403
404 switch( opts.m_NetColorMode )
405 {
406 case NET_COLOR_MODE::ALL: opts.m_NetColorMode = NET_COLOR_MODE::RATSNEST; break;
407 case NET_COLOR_MODE::RATSNEST: opts.m_NetColorMode = NET_COLOR_MODE::OFF; break;
408 case NET_COLOR_MODE::OFF: opts.m_NetColorMode = NET_COLOR_MODE::ALL; break;
409 }
410
411 m_frame->SetDisplayOptions( opts );
412 return 0;
413}
414
415
417{
418 if( !displayOptions().m_ShowGlobalRatsnest )
419 {
421 displayOptions().m_RatsnestMode = RATSNEST_MODE::ALL;
422 }
423 else if( displayOptions().m_RatsnestMode == RATSNEST_MODE::ALL )
424 {
425 displayOptions().m_RatsnestMode = RATSNEST_MODE::VISIBLE;
426 }
427 else
428 {
430 }
431
432 getEditFrame<PCB_EDIT_FRAME>()->SetElementVisibility( LAYER_RATSNEST,
433 displayOptions().m_ShowGlobalRatsnest );
434
435 frame()->OnDisplayOptionsChanged();
436
438 canvas()->Refresh();
439 return 0;
440}
441
442
444{
446
447 return 0;
448}
449
450
452{
453 PCB_BASE_FRAME* editFrame = m_frame;
454 BOARD* brd = board();
455 PCB_LAYER_ID layer = editFrame->GetActiveLayer();
456 bool wraparound = false;
457
458 if( !IsCopperLayer( layer ) )
459 {
460 editFrame->SwitchLayer( B_Cu );
461 return 0;
462 }
463
464 LSET cuMask = LSET::AllCuMask( brd->GetCopperLayerCount() );
465 LSEQ layerStack = cuMask.UIOrder();
466
467 int ii = 0;
468
469 // Find the active layer in list
470 for( ; ii < (int)layerStack.size(); ii++ )
471 {
472 if( layer == layerStack[ii] )
473 break;
474 }
475
476 // Find the next visible layer in list
477 for( ; ii < (int)layerStack.size(); ii++ )
478 {
479 int jj = ii+1;
480
481 if( jj >= (int)layerStack.size() )
482 jj = 0;
483
484 layer = layerStack[jj];
485
486 if( brd->IsLayerVisible( layer ) )
487 break;
488
489 if( jj == 0 ) // the end of list is reached. Try from the beginning
490 {
491 if( wraparound )
492 {
493 wxBell();
494 return 0;
495 }
496 else
497 {
498 wraparound = true;
499 ii = -1;
500 }
501 }
502 }
503
504 wxCHECK( IsCopperLayer( layer ), 0 );
505 editFrame->SwitchLayer( layer );
506
507 return 0;
508}
509
510
512{
513 PCB_BASE_FRAME* editFrame = m_frame;
514 BOARD* brd = board();
515 PCB_LAYER_ID layer = editFrame->GetActiveLayer();
516 bool wraparound = false;
517
518 if( !IsCopperLayer( layer ) )
519 {
520 editFrame->SwitchLayer( F_Cu );
521 return 0;
522 }
523
524 LSET cuMask = LSET::AllCuMask( brd->GetCopperLayerCount() );
525 LSEQ layerStack = cuMask.UIOrder();
526
527 int ii = 0;
528
529 // Find the active layer in list
530 for( ; ii < (int)layerStack.size(); ii++ )
531 {
532 if( layer == layerStack[ii] )
533 break;
534 }
535
536 // Find the previous visible layer in list
537 for( ; ii >= 0; ii-- )
538 {
539 int jj = ii - 1;
540
541 if( jj < 0 )
542 jj = (int)layerStack.size() - 1;
543
544 layer = layerStack[jj];
545
546 if( brd->IsLayerVisible( layer ) )
547 break;
548
549 if( ii == 0 ) // the start of list is reached. Try from the last
550 {
551 if( wraparound )
552 {
553 wxBell();
554 return 0;
555 }
556 else
557 {
558 wraparound = true;
559 ii = 1;
560 }
561 }
562 }
563
564 wxCHECK( IsCopperLayer( layer ), 0 );
565 editFrame->SwitchLayer( layer );
566
567 return 0;
568}
569
570
572{
573 int currentLayer = m_frame->GetActiveLayer();
574 PCB_SCREEN* screen = m_frame->GetScreen();
575
576 if( currentLayer == screen->m_Route_Layer_TOP )
578 else
580
581 return 0;
582}
583
584
585// It'd be nice to share the min/max with the DIALOG_COLOR_PICKER, but those are
586// set in wxFormBuilder.
587#define ALPHA_MIN 0.20
588#define ALPHA_MAX 1.00
589#define ALPHA_STEP 0.05
590
591
593{
595 int currentLayer = m_frame->GetActiveLayer();
596 KIGFX::COLOR4D currentColor = settings->GetColor( currentLayer );
597
598 if( currentColor.a <= ALPHA_MAX - ALPHA_STEP )
599 {
600 currentColor.a += ALPHA_STEP;
601 settings->SetColor( currentLayer, currentColor );
603
605 view->UpdateLayerColor( currentLayer );
606 view->UpdateLayerColor( GetNetnameLayer( currentLayer ) );
607
608 if( IsCopperLayer( currentLayer ) )
609 view->UpdateLayerColor( ZONE_LAYER_FOR( currentLayer ) );
610
612 }
613 else
614 {
615 wxBell();
616 }
617
618 return 0;
619}
620
621
623{
625 int currentLayer = m_frame->GetActiveLayer();
626 KIGFX::COLOR4D currentColor = settings->GetColor( currentLayer );
627
628 if( currentColor.a >= ALPHA_MIN + ALPHA_STEP )
629 {
630 currentColor.a -= ALPHA_STEP;
631 settings->SetColor( currentLayer, currentColor );
633
635 view->UpdateLayerColor( currentLayer );
636 view->UpdateLayerColor( GetNetnameLayer( currentLayer ) );
637
638 if( IsCopperLayer( currentLayer ) )
639 view->UpdateLayerColor( ZONE_LAYER_FOR( currentLayer ) );
640
642 }
643 else
644 {
645 wxBell();
646 }
647
648 return 0;
649}
650
651
653{
654 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame );
655
656 if( !editFrame )
657 return 0;
658
659 LAYER_PAIR_SETTINGS* settings = editFrame->GetLayerPairSettings();
660
661 if( !settings )
662 return 0;
663
664 int currentIndex;
665 std::vector<LAYER_PAIR_INFO> presets = settings->GetEnabledLayerPairs( currentIndex );
666
667 if( presets.size() < 2 )
668 return 0;
669
670 if( currentIndex < 0 )
671 {
672 wxASSERT_MSG( false, "Current layer pair not found in layer settings" );
673 currentIndex = 0;
674 }
675
676 const int nextIndex = ( currentIndex + 1 ) % presets.size();
677 const LAYER_PAIR& nextPair = presets[nextIndex].GetLayerPair();
678
679 settings->SetCurrentLayerPair( nextPair );
680
682 return 0;
683}
684
685
687{
688 if( !Pgm().GetCommonSettings()->m_Input.hotkey_feedback )
689 return 0;
690
691 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame );
692
693 if( !editFrame )
694 return 0;
695
696 LAYER_PAIR_SETTINGS* settings = editFrame->GetLayerPairSettings();
697
698 if( !settings )
699 return 0;
700
701 PCB_LAYER_PRESENTATION layerPresentation( editFrame );
702
703 int currentIndex;
704 std::vector<LAYER_PAIR_INFO> presets = settings->GetEnabledLayerPairs( currentIndex );
705
706 wxArrayString labels;
707 for( const LAYER_PAIR_INFO& layerPairInfo : presets )
708 {
709 wxString label = layerPresentation.getLayerPairName( layerPairInfo.GetLayerPair() );
710
711 if( layerPairInfo.GetName() )
712 {
713 label += wxT( " (" ) + *layerPairInfo.GetName() + wxT( ")" );
714 }
715
716 labels.Add( label );
717 }
718
719 if( !editFrame->GetHotkeyPopup() )
720 editFrame->CreateHotkeyPopup();
721
722 HOTKEY_CYCLE_POPUP* popup = editFrame->GetHotkeyPopup();
723
724 if( popup )
725 {
726 int selection = currentIndex;
727 popup->Popup( _( "Preset Layer Pairs" ), labels, selection );
728 }
729
730 return 0;
731}
732
733
735 EDA_ITEM* originViewItem, const VECTOR2D& aPoint )
736{
737 aFrame->GetDesignSettings().SetGridOrigin( VECTOR2I( aPoint ) );
738 aView->GetGAL()->SetGridOrigin( aPoint );
739 originViewItem->SetPosition( aPoint );
740 aView->MarkDirty();
741 aFrame->OnModify();
742}
743
744
746{
747 VECTOR2D* origin = aEvent.Parameter<VECTOR2D*>();
748
749 if( origin )
750 {
751 // We can't undo the other grid dialog settings, so no sense undoing just the origin
752 DoSetGridOrigin( getView(), m_frame, m_gridOrigin.get(), *origin );
753 delete origin;
754 }
755 else
756 {
757 if( m_isFootprintEditor && !getEditFrame<PCB_BASE_EDIT_FRAME>()->GetModel() )
758 return 0;
759
761
762 if( !picker ) // Happens in footprint wizard
763 return 0;
764
765 // Deactivate other tools; particularly important if another PICKER is currently running
766 Activate();
767
768 picker->SetClickHandler(
769 [this]( const VECTOR2D& pt ) -> bool
770 {
771 m_frame->SaveCopyInUndoList( m_gridOrigin.get(), UNDO_REDO::GRIDORIGIN );
773 return false; // drill origin is a one-shot; don't continue with tool
774 } );
775
777 }
778
779 return 0;
780}
781
782
784{
785 m_frame->SaveCopyInUndoList( m_gridOrigin.get(), UNDO_REDO::GRIDORIGIN );
787 return 0;
788}
789
790
791#define HITTEST_THRESHOLD_PIXELS 5
792
793
795{
797 return 0;
798
800
801 m_pickerItem = nullptr;
803
804 // Deactivate other tools; particularly important if another PICKER is currently running
805 Activate();
806
807 picker->SetCursor( KICURSOR::REMOVE );
808
809 picker->SetClickHandler(
810 [this]( const VECTOR2D& aPosition ) -> bool
811 {
812 if( m_pickerItem )
813 {
815 {
816 m_statusPopup.reset( new STATUS_TEXT_POPUP( m_frame ) );
817 m_statusPopup->SetText( _( "Item locked." ) );
818 m_statusPopup->PopupFor( 2000 );
819 m_statusPopup->Move( KIPLATFORM::UI::GetMousePosition()
820 + wxPoint( 20, 20 ) );
821 return true;
822 }
823
825 selectionTool->UnbrightenItem( m_pickerItem );
826
827 PCB_SELECTION items;
828 items.Add( m_pickerItem );
829
830 EDIT_TOOL* editTool = m_toolMgr->GetTool<EDIT_TOOL>();
831 editTool->DeleteItems( items, false );
832
833 m_pickerItem = nullptr;
834 }
835
836 return true;
837 } );
838
839 picker->SetMotionHandler(
840 [this]( const VECTOR2D& aPos )
841 {
845 GENERAL_COLLECTOR collector;
846 collector.m_Threshold = KiROUND( getView()->ToWorld( HITTEST_THRESHOLD_PIXELS ) );
847
849 collector.Collect( board, GENERAL_COLLECTOR::FootprintItems, aPos, guide );
850 else
851 collector.Collect( board, GENERAL_COLLECTOR::BoardLevelItems, aPos, guide );
852
853 // Remove unselectable items
854 for( int i = collector.GetCount() - 1; i >= 0; --i )
855 {
856 if( !selectionTool->Selectable( collector[ i ] ) )
857 collector.Remove( i );
858 }
859
860 selectionTool->FilterCollectorForHierarchy( collector, false );
861 selectionTool->FilterCollectedItems( collector, false );
862
863 if( collector.GetCount() > 1 )
864 selectionTool->GuessSelectionCandidates( collector, aPos );
865
866 BOARD_ITEM* item = collector.GetCount() == 1 ? collector[ 0 ] : nullptr;
867
868 if( m_pickerItem != item )
869 {
870 if( m_pickerItem )
871 selectionTool->UnbrightenItem( m_pickerItem );
872
873 m_pickerItem = item;
874
875 if( m_pickerItem )
876 selectionTool->BrightenItem( m_pickerItem );
877 }
878 } );
879
880 picker->SetFinalizeHandler(
881 [this]( const int& aFinalState )
882 {
883 if( m_pickerItem )
884 m_toolMgr->GetTool<PCB_SELECTION_TOOL>()->UnbrightenItem( m_pickerItem );
885
886 m_statusPopup.reset();
887
888 // Ensure the cursor gets changed&updated
889 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
891 } );
892
894
895 return 0;
896}
897
898
899static void pasteFootprintItemsToFootprintEditor( FOOTPRINT* aClipFootprint, BOARD* aBoard,
900 std::vector<BOARD_ITEM*>& aPastedItems )
901{
902 FOOTPRINT* editorFootprint = aBoard->GetFirstFootprint();
903
904 aClipFootprint->SetParent( aBoard );
905
906 for( PAD* pad : aClipFootprint->Pads() )
907 {
908 pad->SetParent( editorFootprint );
909 aPastedItems.push_back( pad );
910 }
911
912 aClipFootprint->Pads().clear();
913
914 // Not all items can be added to the current footprint: mandatory fields are already existing
915 // in the current footprint.
916 //
917 for( PCB_FIELD* field : aClipFootprint->GetFields() )
918 {
919 if( field->IsMandatory() )
920 {
921 if( EDA_GROUP* parentGroup = field->GetParentGroup() )
922 parentGroup->RemoveItem( field );
923 }
924 else
925 {
926 PCB_TEXT* text = static_cast<PCB_TEXT*>( field );
927
928 text->SetTextAngle( text->GetTextAngle() - aClipFootprint->GetOrientation() );
929 text->SetTextAngle( text->GetTextAngle() + editorFootprint->GetOrientation() );
930
931 VECTOR2I pos = field->GetFPRelativePosition();
932 field->SetParent( editorFootprint );
933 field->SetFPRelativePosition( pos );
934
935 aPastedItems.push_back( field );
936 }
937 }
938
939 aClipFootprint->GetFields().clear();
940
941 for( BOARD_ITEM* item : aClipFootprint->GraphicalItems() )
942 {
943 if( item->Type() == PCB_TEXT_T )
944 {
945 PCB_TEXT* text = static_cast<PCB_TEXT*>( item );
946
947 text->SetTextAngle( text->GetTextAngle() - aClipFootprint->GetOrientation() );
948 text->SetTextAngle( text->GetTextAngle() + editorFootprint->GetOrientation() );
949 }
950
951 item->Rotate( item->GetPosition(), -aClipFootprint->GetOrientation() );
952 item->Rotate( item->GetPosition(), editorFootprint->GetOrientation() );
953
954 VECTOR2I pos = item->GetFPRelativePosition();
955 item->SetParent( editorFootprint );
956 item->SetFPRelativePosition( pos );
957
958 aPastedItems.push_back( item );
959 }
960
961 aClipFootprint->GraphicalItems().clear();
962
963 for( ZONE* zone : aClipFootprint->Zones() )
964 {
965 zone->SetParent( editorFootprint );
966 aPastedItems.push_back( zone );
967 }
968
969 aClipFootprint->Zones().clear();
970
971 for( PCB_GROUP* group : aClipFootprint->Groups() )
972 {
973 group->SetParent( editorFootprint );
974 aPastedItems.push_back( group );
975 }
976
977 aClipFootprint->Groups().clear();
978}
979
980
981void PCB_CONTROL::pruneItemLayers( std::vector<BOARD_ITEM*>& aItems )
982{
983 // Do not prune items or layers when copying to the FP editor, because all
984 // layers are accepted, even if they are not enabled in the dummy board
985 // This is mainly true for internal copper layers: all are allowed but only one
986 // (In1.cu) is enabled for the GUI.
988 return;
989
990 LSET enabledLayers = board()->GetEnabledLayers();
991 std::vector<BOARD_ITEM*> returnItems;
992 bool fpItemDeleted = false;
993
994 for( BOARD_ITEM* item : aItems )
995 {
996 if( item->Type() == PCB_FOOTPRINT_T )
997 {
998 FOOTPRINT* fp = static_cast<FOOTPRINT*>( item );
999
1000 // Items living in a parent footprint are never removed, even if their
1001 // layer does not exist in the board editor
1002 // Otherwise the parent footprint could be seriously broken especially
1003 // if some layers are later re-enabled.
1004 // Moreover a fp lives in a fp library, that does not know the enabled
1005 // layers of a given board, so fp items are just ignored when on not
1006 // enabled layers in board editor
1007 returnItems.push_back( fp );
1008 }
1009 else if( item->Type() == PCB_GROUP_T || item->Type() == PCB_GENERATOR_T )
1010 {
1011 returnItems.push_back( item );
1012 }
1013 else
1014 {
1015 LSET allowed = item->GetLayerSet() & enabledLayers;
1016 bool item_valid = true;
1017
1018 // Ensure, for vias, the top and bottom layers are compatible with
1019 // the current board copper layers.
1020 // Otherwise they must be skipped, even is one layer is valid
1021 if( item->Type() == PCB_VIA_T )
1022 item_valid = static_cast<PCB_VIA*>( item )->HasValidLayerPair(
1024
1025 if( allowed.any() && item_valid )
1026 {
1027 item->SetLayerSet( allowed );
1028 returnItems.push_back( item );
1029 }
1030 else
1031 {
1032 if( EDA_GROUP* parentGroup = item->GetParentGroup() )
1033 parentGroup->RemoveItem( item );
1034 }
1035 }
1036 }
1037
1038 if( ( returnItems.size() < aItems.size() ) || fpItemDeleted )
1039 {
1040 DisplayError( m_frame, _( "Warning: some pasted items were on layers which are not "
1041 "present in the current board.\n"
1042 "These items could not be pasted.\n" ) );
1043 }
1044
1045 aItems = returnItems;
1046}
1047
1048
1049int PCB_CONTROL::Paste( const TOOL_EVENT& aEvent )
1050{
1051 // The viewer frames cannot paste
1052 if( !frame()->IsType( FRAME_FOOTPRINT_EDITOR ) && !frame()->IsType( FRAME_PCB_EDITOR ) )
1053 return 0;
1054
1055 bool isFootprintEditor = m_isFootprintEditor || frame()->IsType( FRAME_FOOTPRINT_EDITOR );
1056 // The clipboard can contain two different things, an entire kicad_pcb or a single footprint
1057 if( isFootprintEditor && ( !board() || !footprint() ) )
1058 return 0;
1059
1060 // We should never get here if a modal dialog is up... but we do on MacOS.
1061 // https://gitlab.com/kicad/code/kicad/-/issues/18912
1062#ifdef __WXMAC__
1063 if( wxDialog::OSXHasModalDialogsOpen() )
1064 {
1065 wxBell();
1066 return 0;
1067 }
1068#endif
1069
1070 BOARD_COMMIT commit( frame() );
1071
1072 CLIPBOARD_IO pi;
1073 BOARD_ITEM* clipItem = pi.Parse();
1074
1075 if( !clipItem )
1076 {
1077 // When the clipboard doesn't parse, create a PCB item with the clipboard contents
1078 std::vector<BOARD_ITEM*> newItems;
1079
1080 if( std::unique_ptr<wxImage> clipImg = GetImageFromClipboard() )
1081 {
1082 auto refImg = std::make_unique<PCB_REFERENCE_IMAGE>( m_frame->GetModel() );
1083
1084 if( refImg->GetReferenceImage().SetImage( *clipImg ) )
1085 newItems.push_back( refImg.release() );
1086 }
1087 else
1088 {
1089 const wxString clipText = GetClipboardUTF8();
1090
1091 if( clipText.empty() )
1092 return 0;
1093
1094 std::unique_ptr<PCB_TEXT> item;
1095
1096 item = std::make_unique<PCB_TEXT>( m_frame->GetModel() );
1097 item->SetText( clipText );
1098
1099 newItems.push_back( item.release() );
1100 }
1101
1102 bool cancelled = !placeBoardItems( &commit, newItems, true, false, false );
1103
1104 if( cancelled )
1105 commit.Revert();
1106 else
1107 commit.Push( _( "Paste Text" ) );
1108 return 0;
1109 }
1110
1111 // If we get here, we have a parsed board/FP to paste
1112
1113 PASTE_MODE mode = PASTE_MODE::KEEP_ANNOTATIONS;
1114 bool clear_nets = false;
1115 const wxString defaultRef = wxT( "REF**" );
1116
1117 if( aEvent.IsAction( &ACTIONS::pasteSpecial ) )
1118 {
1119 DIALOG_PASTE_SPECIAL dlg( m_frame, &mode, defaultRef );
1120
1121 if( clipItem->Type() != PCB_T )
1122 dlg.HideClearNets();
1123
1124 if( dlg.ShowModal() == wxID_CANCEL )
1125 return 0;
1126
1127 clear_nets = dlg.GetClearNets();
1128 }
1129
1130 if( clipItem->Type() == PCB_T )
1131 {
1132 BOARD* clipBoard = static_cast<BOARD*>( clipItem );
1133
1134 if( isFootprintEditor || clear_nets )
1135 {
1136 for( BOARD_CONNECTED_ITEM* item : clipBoard->AllConnectedItems() )
1137 item->SetNet( NETINFO_LIST::OrphanedItem() );
1138 }
1139 else
1140 {
1141 clipBoard->MapNets( m_frame->GetBoard() );
1142 }
1143 }
1144
1145 bool cancelled = false;
1146
1147 switch( clipItem->Type() )
1148 {
1149 case PCB_T:
1150 {
1151 BOARD* clipBoard = static_cast<BOARD*>( clipItem );
1152
1153 if( isFootprintEditor )
1154 {
1155 FOOTPRINT* editorFootprint = board()->GetFirstFootprint();
1156 std::vector<BOARD_ITEM*> pastedItems;
1157
1158 for( PCB_GROUP* group : clipBoard->Groups() )
1159 {
1160 group->SetParent( editorFootprint );
1161 pastedItems.push_back( group );
1162 }
1163
1164 clipBoard->RemoveAll( { PCB_GROUP_T } );
1165
1166 for( FOOTPRINT* clipFootprint : clipBoard->Footprints() )
1167 pasteFootprintItemsToFootprintEditor( clipFootprint, board(), pastedItems );
1168
1169 for( BOARD_ITEM* clipDrawItem : clipBoard->Drawings() )
1170 {
1171 switch( clipDrawItem->Type() )
1172 {
1173 case PCB_TEXT_T:
1174 case PCB_TEXTBOX_T:
1175 case PCB_TABLE_T:
1176 case PCB_SHAPE_T:
1177 case PCB_DIM_ALIGNED_T:
1178 case PCB_DIM_CENTER_T:
1179 case PCB_DIM_LEADER_T:
1181 case PCB_DIM_RADIAL_T:
1182 clipDrawItem->SetParent( editorFootprint );
1183 pastedItems.push_back( clipDrawItem );
1184 break;
1185
1186 default:
1187 // Everything we *didn't* put into pastedItems is going to get nuked, so
1188 // make sure it's not still included in its parent group.
1189 if( EDA_GROUP* parentGroup = clipDrawItem->GetParentGroup() )
1190 parentGroup->RemoveItem( clipDrawItem );
1191
1192 break;
1193 }
1194 }
1195
1196 // NB: PCB_SHAPE_T actually removes everything in Drawings() (including PCB_TEXTs,
1197 // PCB_TABLES, dimensions, etc.), not just PCB_SHAPEs.)
1198 clipBoard->RemoveAll( { PCB_SHAPE_T } );
1199
1200 clipBoard->Visit(
1201 [&]( EDA_ITEM* item, void* testData )
1202 {
1203 // Anything still on the clipboard didn't get copied and needs to be
1204 // removed from the pasted groups.
1205 BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( item );
1206 EDA_GROUP* parentGroup = boardItem->GetParentGroup();
1207
1208 if( parentGroup )
1209 parentGroup->RemoveItem( boardItem );
1210
1211 return INSPECT_RESULT::CONTINUE;
1212 },
1214
1215 delete clipBoard;
1216
1217 pruneItemLayers( pastedItems );
1218
1219 cancelled = !placeBoardItems( &commit, pastedItems, true, true,
1220 mode == PASTE_MODE::UNIQUE_ANNOTATIONS );
1221 }
1222 else // isBoardEditor
1223 {
1224 // Fixup footprint component classes
1225 for( FOOTPRINT* clipFootprint : clipBoard->Footprints() )
1226 {
1227 clipFootprint->ResolveComponentClassNames(
1228 board(), clipFootprint->GetTransientComponentClassNames() );
1229 clipFootprint->ClearTransientComponentClassNames();
1230 }
1231
1232 if( mode == PASTE_MODE::REMOVE_ANNOTATIONS )
1233 {
1234 for( FOOTPRINT* clipFootprint : clipBoard->Footprints() )
1235 clipFootprint->SetReference( defaultRef );
1236 }
1237
1238 cancelled = !placeBoardItems( &commit, clipBoard, true,
1239 mode == PASTE_MODE::UNIQUE_ANNOTATIONS );
1240 }
1241
1242 break;
1243 }
1244
1245 case PCB_FOOTPRINT_T:
1246 {
1247 FOOTPRINT* clipFootprint = static_cast<FOOTPRINT*>( clipItem );
1248 std::vector<BOARD_ITEM*> pastedItems;
1249
1250 if( isFootprintEditor )
1251 {
1252 pasteFootprintItemsToFootprintEditor( clipFootprint, board(), pastedItems );
1253 delete clipFootprint;
1254 }
1255 else
1256 {
1257 if( mode == PASTE_MODE::REMOVE_ANNOTATIONS )
1258 clipFootprint->SetReference( defaultRef );
1259
1260 clipFootprint->SetParent( board() );
1261 clipFootprint->ResolveComponentClassNames(
1262 board(), clipFootprint->GetTransientComponentClassNames() );
1263 clipFootprint->ClearTransientComponentClassNames();
1264 pastedItems.push_back( clipFootprint );
1265 }
1266
1267 pruneItemLayers( pastedItems );
1268
1269 cancelled = !placeBoardItems( &commit, pastedItems, true, true,
1270 mode == PASTE_MODE::UNIQUE_ANNOTATIONS );
1271 break;
1272 }
1273
1274 default:
1275 m_frame->DisplayToolMsg( _( "Invalid clipboard contents" ) );
1276 break;
1277 }
1278
1279 if( cancelled )
1280 commit.Revert();
1281 else
1282 commit.Push( _( "Paste" ) );
1283
1284 return 1;
1285}
1286
1287
1289{
1290 wxString fileName;
1291
1292 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame );
1293
1294 if( !editFrame )
1295 return 1;
1296
1297 // Pick a file to append
1298 if( !AskLoadBoardFileName( editFrame, &fileName, KICTL_KICAD_ONLY ) )
1299 return 1;
1300
1301 PCB_IO_MGR::PCB_FILE_T pluginType =
1303 IO_RELEASER<PCB_IO> pi( PCB_IO_MGR::PluginFind( pluginType ) );
1304
1305 if( !pi )
1306 return 1;
1307
1308 return AppendBoard( *pi, fileName );
1309}
1310
1311
1313{
1314 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame );
1315
1316 if( !editFrame )
1317 return 1;
1318
1319 DESIGN_BLOCK* designBlock = nullptr;
1320
1321 if( !editFrame->GetDesignBlockPane()->GetSelectedLibId().IsValid() )
1322 return 1;
1323
1324
1325 designBlock = editFrame->GetDesignBlockPane()->GetDesignBlock( editFrame->GetDesignBlockPane()->GetSelectedLibId(),
1326 true, true );
1327
1328 if( !designBlock || designBlock->GetBoardFile().IsEmpty() )
1329 return 1;
1330
1331
1333 IO_RELEASER<PCB_IO> pi( PCB_IO_MGR::PluginFind( pluginType ) );
1334
1335 if( !pi )
1336 return 1;
1337
1338 bool repeatPlacement = false;
1339
1340 if( APP_SETTINGS_BASE* cfg = editFrame->config() )
1341 repeatPlacement = cfg->m_DesignBlockChooserPanel.repeated_placement;
1342
1343 int ret = 0;
1344
1345 do
1346 {
1347 ret = AppendBoard( *pi, designBlock->GetBoardFile() );
1348 } while( repeatPlacement && ret == 0 );
1349
1350 return ret;
1351}
1352
1353
1354template<typename T>
1355static void moveUnflaggedItems( const std::deque<T>& aList, std::vector<BOARD_ITEM*>& aTarget,
1356 bool aIsNew )
1357{
1358 std::copy_if( aList.begin(), aList.end(), std::back_inserter( aTarget ),
1359 [aIsNew]( T aItem )
1360 {
1361 bool doCopy = ( aItem->GetFlags() & SKIP_STRUCT ) == 0;
1362
1363 aItem->ClearFlags( SKIP_STRUCT );
1364 aItem->SetFlags( aIsNew ? IS_NEW : 0 );
1365
1366 return doCopy;
1367 } );
1368}
1369
1370
1371template<typename T>
1372static void moveUnflaggedItems( const std::vector<T>& aList, std::vector<BOARD_ITEM*>& aTarget,
1373 bool aIsNew )
1374{
1375 std::copy_if( aList.begin(), aList.end(), std::back_inserter( aTarget ),
1376 [aIsNew]( T aItem )
1377 {
1378 bool doCopy = ( aItem->GetFlags() & SKIP_STRUCT ) == 0;
1379
1380 aItem->ClearFlags( SKIP_STRUCT );
1381 aItem->SetFlags( aIsNew ? IS_NEW : 0 );
1382
1383 return doCopy;
1384 } );
1385}
1386
1387
1388bool PCB_CONTROL::placeBoardItems( BOARD_COMMIT* aCommit, BOARD* aBoard, bool aAnchorAtOrigin,
1389 bool aReannotateDuplicates )
1390{
1391 // items are new if the current board is not the board source
1392 bool isNew = board() != aBoard;
1393 std::vector<BOARD_ITEM*> items;
1394
1395 moveUnflaggedItems( aBoard->Tracks(), items, isNew );
1396 moveUnflaggedItems( aBoard->Footprints(), items, isNew );
1397 moveUnflaggedItems( aBoard->Drawings(), items, isNew );
1398 moveUnflaggedItems( aBoard->Zones(), items, isNew );
1399
1400 // Subtlety: When selecting a group via the mouse,
1401 // PCB_SELECTION_TOOL::highlightInternal runs, which does a SetSelected() on all
1402 // descendants. In PCB_CONTROL::placeBoardItems, below, we skip that and
1403 // mark items non-recursively. That works because the saving of the
1404 // selection created aBoard that has the group and all descendants in it.
1405 moveUnflaggedItems( aBoard->Groups(), items, isNew );
1406
1407 moveUnflaggedItems( aBoard->Generators(), items, isNew );
1408
1409 if( isNew )
1410 aBoard->RemoveAll();
1411
1412 // Reparent before calling pruneItemLayers, as SetLayer can have a dependence on the
1413 // item's parent board being set correctly.
1414 if( isNew )
1415 {
1416 for( BOARD_ITEM* item : items )
1417 item->SetParent( board() );
1418 }
1419
1420 pruneItemLayers( items );
1421
1422 return placeBoardItems( aCommit, items, isNew, aAnchorAtOrigin, aReannotateDuplicates );
1423}
1424
1425
1426bool PCB_CONTROL::placeBoardItems( BOARD_COMMIT* aCommit, std::vector<BOARD_ITEM*>& aItems,
1427 bool aIsNew, bool aAnchorAtOrigin, bool aReannotateDuplicates )
1428{
1430
1432
1433 std::vector<BOARD_ITEM*> itemsToSel;
1434 itemsToSel.reserve( aItems.size() );
1435
1436 for( BOARD_ITEM* item : aItems )
1437 {
1438 if( aIsNew )
1439 {
1440 const_cast<KIID&>( item->m_Uuid ) = KIID();
1441
1442 item->RunOnChildren(
1443 []( BOARD_ITEM* aChild )
1444 {
1445 const_cast<KIID&>( aChild->m_Uuid ) = KIID();
1446 },
1447 RECURSE_MODE::RECURSE );
1448
1449 // Even though BOARD_COMMIT::Push() will add any new items to the group, we're
1450 // going to run PCB_ACTIONS::move first, and the move tool will throw out any
1451 // items that aren't in the entered group.
1452 if( selectionTool->GetEnteredGroup() && !item->GetParentGroup() )
1453 selectionTool->GetEnteredGroup()->AddItem( item );
1454
1455 item->SetParent( board() );
1456 }
1457
1458 // Update item attributes if needed
1459 if( BaseType( item->Type() ) == PCB_DIMENSION_T )
1460 {
1461 static_cast<PCB_DIMENSION_BASE*>( item )->UpdateUnits();
1462 }
1463 else if( item->Type() == PCB_FOOTPRINT_T )
1464 {
1465 FOOTPRINT* footprint = static_cast<FOOTPRINT*>( item );
1466
1467 // Update the footprint path with the new KIID path if the footprint is new
1468 if( aIsNew )
1470
1471 for( BOARD_ITEM* dwg : footprint->GraphicalItems() )
1472 {
1473 if( BaseType( dwg->Type() ) == PCB_DIMENSION_T )
1474 static_cast<PCB_DIMENSION_BASE*>( dwg )->UpdateUnits();
1475 }
1476 }
1477
1478 // We only need to add the items that aren't inside a group currently selected
1479 // to the selection. If an item is inside a group and that group is selected,
1480 // then the selection tool will select it for us.
1481 if( !item->GetParentGroup() || !alg::contains( aItems, item->GetParentGroup()->AsEdaItem() ) )
1482 itemsToSel.push_back( item );
1483 }
1484
1485 // Select the items that should be selected
1486 EDA_ITEMS toSel( itemsToSel.begin(), itemsToSel.end() );
1488
1489 // Reannotate duplicate footprints (make sense only in board editor )
1490 if( aReannotateDuplicates && m_isBoardEditor )
1491 m_toolMgr->GetTool<BOARD_REANNOTATE_TOOL>()->ReannotateDuplicatesInSelection();
1492
1493 for( BOARD_ITEM* item : aItems )
1494 {
1495 if( aIsNew )
1496 aCommit->Add( item );
1497 else
1498 aCommit->Added( item );
1499 }
1500
1501 PCB_SELECTION& selection = selectionTool->GetSelection();
1502
1503 if( selection.Size() > 0 )
1504 {
1505 if( aAnchorAtOrigin )
1506 {
1508 }
1509 else
1510 {
1511 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( selection.GetTopLeftItem() );
1513 }
1514
1515 getViewControls()->SetCursorPosition( getViewControls()->GetMousePosition(), false );
1516
1518
1520 }
1521
1522 return true;
1523}
1524
1525
1526int PCB_CONTROL::AppendBoard( PCB_IO& pi, const wxString& fileName )
1527{
1528 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame );
1529
1530 if( !editFrame )
1531 return 1;
1532
1533 BOARD* brd = board();
1534
1535 if( !brd )
1536 return 1;
1537
1538 BOARD_COMMIT commit( editFrame );
1539
1540 // Mark existing items, in order to know what are the new items so we can select only
1541 // the new items after loading
1542 for( PCB_TRACK* track : brd->Tracks() )
1543 track->SetFlags( SKIP_STRUCT );
1544
1545 for( FOOTPRINT* footprint : brd->Footprints() )
1547
1548 for( PCB_GROUP* group : brd->Groups() )
1549 group->SetFlags( SKIP_STRUCT );
1550
1551 for( BOARD_ITEM* drawing : brd->Drawings() )
1552 drawing->SetFlags( SKIP_STRUCT );
1553
1554 for( ZONE* zone : brd->Zones() )
1555 zone->SetFlags( SKIP_STRUCT );
1556
1557 for( PCB_GENERATOR* generator : brd->Generators() )
1558 generator->SetFlags( SKIP_STRUCT );
1559
1560 std::map<wxString, wxString> oldProperties = brd->GetProperties();
1561 std::map<wxString, wxString> newProperties;
1562
1563 PAGE_INFO oldPageInfo = brd->GetPageSettings();
1564 TITLE_BLOCK oldTitleBlock = brd->GetTitleBlock();
1565
1566 // Keep also the count of copper layers, to adjust if necessary
1567 int initialCopperLayerCount = brd->GetCopperLayerCount();
1568 LSET initialEnabledLayers = brd->GetEnabledLayers();
1569
1570 // Load the data
1571 try
1572 {
1573 std::map<std::string, UTF8> props;
1574
1575 // PCB_IO_EAGLE can use this info to center the BOARD, but it does not yet.
1576
1577 props["page_width"] = std::to_string( editFrame->GetPageSizeIU().x );
1578 props["page_height"] = std::to_string( editFrame->GetPageSizeIU().y );
1579
1581 [&]( wxString aTitle, int aIcon, wxString aMessage, wxString aAction ) -> bool
1582 {
1583 KIDIALOG dlg( editFrame, aMessage, aTitle, wxOK | wxCANCEL | aIcon );
1584
1585 if( !aAction.IsEmpty() )
1586 dlg.SetOKLabel( aAction );
1587
1588 dlg.DoNotShowCheckbox( aMessage, 0 );
1589
1590 return dlg.ShowModal() == wxID_OK;
1591 } );
1592
1593 WX_PROGRESS_REPORTER progressReporter( editFrame, _( "Loading PCB" ), 1 );
1594
1595 editFrame->GetDesignSettings().m_NetSettings->ClearNetclasses();
1596 pi.SetProgressReporter( &progressReporter );
1597 pi.LoadBoard( fileName, brd, &props, nullptr );
1598 }
1599 catch( const IO_ERROR& ioe )
1600 {
1601 DisplayErrorMessage( editFrame, _( "Error loading board." ), ioe.What() );
1602
1603 return 0;
1604 }
1605
1606 newProperties = brd->GetProperties();
1607
1608 for( const std::pair<const wxString, wxString>& prop : oldProperties )
1609 newProperties[ prop.first ] = prop.second;
1610
1611 brd->SetProperties( newProperties );
1612
1613 brd->SetPageSettings( oldPageInfo );
1614 brd->SetTitleBlock( oldTitleBlock );
1615
1616 // rebuild nets and ratsnest before any use of nets
1617 brd->BuildListOfNets();
1618 brd->SynchronizeNetsAndNetClasses( true );
1619 brd->BuildConnectivity();
1620
1621 // Synchronize layers
1622 // we should not ask PLUGINs to do these items:
1623 int copperLayerCount = brd->GetCopperLayerCount();
1624
1625 if( copperLayerCount > initialCopperLayerCount )
1626 brd->SetCopperLayerCount( copperLayerCount );
1627
1628 // Enable all used layers, and make them visible:
1629 LSET enabledLayers = brd->GetEnabledLayers();
1630 enabledLayers |= initialEnabledLayers;
1631 brd->SetEnabledLayers( enabledLayers );
1632 brd->SetVisibleLayers( enabledLayers );
1633
1634 int ret = 0;
1635
1636 bool placeAsGroup = editFrame->config() ? editFrame->config()->m_DesignBlockChooserPanel.place_as_sheet : false;
1637
1638 if( placeBoardItems( &commit, brd, false, false /* Don't reannotate dupes on Append Board */ ) )
1639 {
1640 commit.Push( _( "Append Board" ) );
1641
1642 if( placeAsGroup )
1643 {
1644 BOARD_COMMIT grpCommit( m_toolMgr );
1645 PCB_GROUP* group = new PCB_GROUP( brd );
1646
1647 // Get the selection tool selection
1650
1651 for( EDA_ITEM* eda_item : selection )
1652 {
1653 if( eda_item->IsBOARD_ITEM() )
1654 {
1655 if( static_cast<BOARD_ITEM*>( eda_item )->IsLocked() )
1656 group->SetLocked( true );
1657 }
1658 }
1659
1660 grpCommit.Add( group );
1661
1662 for( EDA_ITEM* eda_item : selection )
1663 {
1664 if( eda_item->IsBOARD_ITEM() && !static_cast<BOARD_ITEM*>( eda_item )->GetParentFootprint() )
1665 {
1666 grpCommit.Stage( static_cast<BOARD_ITEM*>( eda_item ), CHT_GROUP );
1667 }
1668 }
1669
1670 grpCommit.Push( _( "Group Items" ), APPEND_UNDO );
1671
1672 selTool->ClearSelection();
1673 selTool->select( group );
1674
1676 m_frame->OnModify();
1677 m_frame->Refresh();
1678 }
1679
1680 ret = 0;
1681 }
1682 else
1683 {
1684 commit.Revert();
1685 ret = 1;
1686 }
1687
1688 // Refresh the UI for the updated board properties
1689 editFrame->GetAppearancePanel()->OnBoardChanged();
1690
1691 return ret;
1692}
1693
1694
1695int PCB_CONTROL::Undo( const TOOL_EVENT& aEvent )
1696{
1697 PCB_BASE_EDIT_FRAME* editFrame = dynamic_cast<PCB_BASE_EDIT_FRAME*>( m_frame );
1698 wxCommandEvent dummy;
1699
1700 if( editFrame )
1701 editFrame->RestoreCopyFromUndoList( dummy );
1702
1703 return 0;
1704}
1705
1706
1707int PCB_CONTROL::Redo( const TOOL_EVENT& aEvent )
1708{
1709 PCB_BASE_EDIT_FRAME* editFrame = dynamic_cast<PCB_BASE_EDIT_FRAME*>( m_frame );
1710 wxCommandEvent dummy;
1711
1712 if( editFrame )
1713 editFrame->RestoreCopyFromRedoList( dummy );
1714
1715 return 0;
1716}
1717
1718
1720{
1724 bool& snapMode = settings.allLayers;
1725
1727 snapMode = false;
1728 else if( aEvent.IsAction( &PCB_ACTIONS::magneticSnapAllLayers ) )
1729 snapMode = true;
1730 else
1731 snapMode = !snapMode;
1732
1734
1735 return 0;
1736}
1737
1738
1740{
1741 if( !Pgm().GetCommonSettings()->m_Input.hotkey_feedback )
1742 return 0;
1743
1744 wxArrayString labels;
1745 labels.Add( _( "Active Layer" ) );
1746 labels.Add( _( "All Layers" ) );
1747
1748 if( !m_frame->GetHotkeyPopup() )
1750
1752
1756
1757 if( popup )
1758 popup->Popup( _( "Object Snapping" ), labels, static_cast<int>( settings.allLayers ) );
1759
1760 return 0;
1761}
1762
1763
1765{
1767 ROUTER_TOOL* routerTool = m_toolMgr->GetTool<ROUTER_TOOL>();
1768 PCB_SELECTION& selection = selTool->GetSelection();
1769 PCB_EDIT_FRAME* pcbFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame );
1770 std::shared_ptr<DRC_ENGINE> drcEngine = m_frame->GetBoard()->GetDesignSettings().m_DRCEngine;
1771 DRC_CONSTRAINT constraint;
1772
1773 std::vector<MSG_PANEL_ITEM> msgItems;
1774
1775 if( routerTool && routerTool->RoutingInProgress() )
1776 {
1777 routerTool->UpdateMessagePanel();
1778 return 0;
1779 }
1780
1781 if( !pcbFrame && !m_frame->GetModel() )
1782 return 0;
1783
1784 if( selection.Empty() )
1785 {
1786 if( !pcbFrame )
1787 {
1788 FOOTPRINT* fp = static_cast<FOOTPRINT*>( m_frame->GetModel() );
1789 fp->GetMsgPanelInfo( m_frame, msgItems );
1790 }
1791 else
1792 {
1794 }
1795 }
1796 else if( selection.GetSize() == 1 )
1797 {
1798 EDA_ITEM* item = selection.Front();
1799
1800 if( std::optional<wxString> uuid = GetMsgPanelDisplayUuid( item->m_Uuid ) )
1801 msgItems.emplace_back( _( "UUID" ), *uuid );
1802
1803 item->GetMsgPanelInfo( m_frame, msgItems );
1804
1805 PCB_TRACK* track = dynamic_cast<PCB_TRACK*>( item );
1806 NETINFO_ITEM* net = track ? track->GetNet() : nullptr;
1807 NETINFO_ITEM* coupledNet = net ? m_frame->GetBoard()->DpCoupledNet( net ) : nullptr;
1808
1809 if( coupledNet )
1810 {
1811 SEG trackSeg( track->GetStart(), track->GetEnd() );
1812 PCB_TRACK* coupledItem = nullptr;
1813 SEG::ecoord closestDist_sq = VECTOR2I::ECOORD_MAX;
1814
1815 for( PCB_TRACK* candidate : m_frame->GetBoard()->Tracks() )
1816 {
1817 if( candidate->GetNet() != coupledNet )
1818 continue;
1819
1820 SEG::ecoord dist_sq = trackSeg.SquaredDistance( SEG( candidate->GetStart(),
1821 candidate->GetEnd() ) );
1822
1823 if( !coupledItem || dist_sq < closestDist_sq )
1824 {
1825 coupledItem = candidate;
1826 closestDist_sq = dist_sq;
1827 }
1828 }
1829
1830 constraint = drcEngine->EvalRules( DIFF_PAIR_GAP_CONSTRAINT, track, coupledItem,
1831 track->GetLayer() );
1832
1833 wxString msg = m_frame->MessageTextFromMinOptMax( constraint.Value() );
1834
1835 if( !msg.IsEmpty() )
1836 {
1837 msgItems.emplace_back( wxString::Format( _( "DP Gap Constraints: %s" ), msg ),
1838 wxString::Format( _( "(from %s)" ), constraint.GetName() ) );
1839 }
1840
1841 constraint = drcEngine->EvalRules( MAX_UNCOUPLED_CONSTRAINT, track,
1842 coupledItem, track->GetLayer() );
1843
1844 if( constraint.Value().HasMax() )
1845 {
1846 msg = m_frame->MessageTextFromValue( constraint.Value().Max() );
1847 msgItems.emplace_back( wxString::Format( _( "DP Max Uncoupled-length: %s" ), msg ),
1848 wxString::Format( _( "(from %s)" ), constraint.GetName() ) );
1849 }
1850 }
1851 }
1852 else if( pcbFrame && selection.GetSize() == 2 )
1853 {
1854 // Pair selection broken into multiple, optional data, starting with the selected item
1855 // names
1856
1857 BOARD_ITEM* a = static_cast<BOARD_ITEM*>( selection[0] );
1858 BOARD_ITEM* b = static_cast<BOARD_ITEM*>( selection[1] );
1859
1860 msgItems.emplace_back( MSG_PANEL_ITEM( a->GetItemDescription( m_frame, false ),
1861 b->GetItemDescription( m_frame, false ) ) );
1862
1863 BOARD_CONNECTED_ITEM* a_conn = dyn_cast<BOARD_CONNECTED_ITEM*>( a );
1864 BOARD_CONNECTED_ITEM* b_conn = dyn_cast<BOARD_CONNECTED_ITEM*>( b );
1865
1866 if( a_conn && b_conn )
1867 {
1868 LSET overlap = a_conn->GetLayerSet() & b_conn->GetLayerSet() & LSET::AllCuMask();
1869 int a_netcode = a_conn->GetNetCode();
1870 int b_netcode = b_conn->GetNetCode();
1871
1872 if( overlap.count() > 0
1873 && ( a_netcode != b_netcode || a_netcode < 0 || b_netcode < 0 ) )
1874 {
1875 PCB_LAYER_ID layer = overlap.CuStack().front();
1876
1877 constraint = drcEngine->EvalRules( CLEARANCE_CONSTRAINT, a, b, layer );
1878 msgItems.emplace_back( _( "Resolved Clearance" ),
1879 m_frame->MessageTextFromValue( constraint.m_Value.Min() ) );
1880
1881 std::shared_ptr<SHAPE> a_shape( a_conn->GetEffectiveShape( layer ) );
1882 std::shared_ptr<SHAPE> b_shape( b_conn->GetEffectiveShape( layer ) );
1883
1884 int actual_clearance = a_shape->GetClearance( b_shape.get() );
1885
1886 if( actual_clearance > -1 && actual_clearance < std::numeric_limits<int>::max() )
1887 {
1888 msgItems.emplace_back( _( "Actual Clearance" ),
1889 m_frame->MessageTextFromValue( actual_clearance ) );
1890 }
1891 }
1892 }
1893
1894 if( ( a->HasHole() || b->HasHole() ) )
1895 {
1898
1899 if( b->IsOnLayer( active ) && IsCopperLayer( active ) )
1900 layer = active;
1901 else if( b->HasHole() && a->IsOnLayer( active ) && IsCopperLayer( active ) )
1902 layer = active;
1903 else if( a->HasHole() && b->IsOnCopperLayer() )
1904 layer = b->GetLayer();
1905 else if( b->HasHole() && a->IsOnCopperLayer() )
1906 layer = a->GetLayer();
1907
1908 if( IsCopperLayer( layer ) )
1909 {
1910 int actual = std::numeric_limits<int>::max();
1911
1912 if( a->HasHole() && b->IsOnCopperLayer() )
1913 {
1914 std::shared_ptr<SHAPE_SEGMENT> hole = a->GetEffectiveHoleShape();
1915 std::shared_ptr<SHAPE> other( b->GetEffectiveShape( layer ) );
1916
1917 actual = std::min( actual, hole->GetClearance( other.get() ) );
1918 }
1919
1920 if( b->HasHole() && a->IsOnCopperLayer() )
1921 {
1922 std::shared_ptr<SHAPE_SEGMENT> hole = b->GetEffectiveHoleShape();
1923 std::shared_ptr<SHAPE> other( a->GetEffectiveShape( layer ) );
1924
1925 actual = std::min( actual, hole->GetClearance( other.get() ) );
1926 }
1927
1928 if( actual < std::numeric_limits<int>::max() )
1929 {
1930 constraint = drcEngine->EvalRules( HOLE_CLEARANCE_CONSTRAINT, a, b, layer );
1931 msgItems.emplace_back( _( "Resolved Hole Clearance" ),
1932 m_frame->MessageTextFromValue( constraint.m_Value.Min() ) );
1933
1934 if( actual > -1 && actual < std::numeric_limits<int>::max() )
1935 {
1936 msgItems.emplace_back( _( "Actual Hole Clearance" ),
1938 }
1939 }
1940 }
1941 }
1942
1943 for( PCB_LAYER_ID edgeLayer : { Edge_Cuts, Margin } )
1944 {
1947
1948 if( a->IsOnLayer( edgeLayer ) && b->Type() != PCB_FOOTPRINT_T )
1949 {
1950 if( b->IsOnLayer( active ) && IsCopperLayer( active ) )
1951 layer = active;
1952 else if( IsCopperLayer( b->GetLayer() ) )
1953 layer = b->GetLayer();
1954 }
1955 else if( b->IsOnLayer( edgeLayer ) && a->Type() != PCB_FOOTPRINT_T )
1956 {
1957 if( a->IsOnLayer( active ) && IsCopperLayer( active ) )
1958 layer = active;
1959 else if( IsCopperLayer( a->GetLayer() ) )
1960 layer = a->GetLayer();
1961 }
1962
1963 if( layer >= 0 )
1964 {
1965 constraint = drcEngine->EvalRules( EDGE_CLEARANCE_CONSTRAINT, a, b, layer );
1966
1967 if( edgeLayer == Edge_Cuts )
1968 {
1969 msgItems.emplace_back( _( "Resolved Edge Clearance" ),
1970 m_frame->MessageTextFromValue( constraint.m_Value.Min() ) );
1971 }
1972 else
1973 {
1974 msgItems.emplace_back( _( "Resolved Margin Clearance" ),
1975 m_frame->MessageTextFromValue( constraint.m_Value.Min() ) );
1976 }
1977 }
1978 }
1979 }
1980
1981 if( selection.GetSize() )
1982 {
1983 if( msgItems.empty() )
1984 {
1985 msgItems.emplace_back( _( "Selected Items" ),
1986 wxString::Format( wxT( "%d" ), selection.GetSize() ) );
1987
1988 if( m_isBoardEditor )
1989 {
1990 std::set<wxString> netNames;
1991 std::set<wxString> netClasses;
1992
1993 for( EDA_ITEM* item : selection )
1994 {
1995 if( BOARD_CONNECTED_ITEM* bci = dynamic_cast<BOARD_CONNECTED_ITEM*>( item ) )
1996 {
1997 netNames.insert( UnescapeString( bci->GetNetname() ) );
1998 netClasses.insert( UnescapeString( bci->GetEffectiveNetClass()->GetHumanReadableName() ) );
1999
2000 if( netNames.size() > 1 && netClasses.size() > 1 )
2001 break;
2002 }
2003 }
2004
2005 if( netNames.size() == 1 )
2006 msgItems.emplace_back( _( "Net" ), *netNames.begin() );
2007
2008 if( netClasses.size() == 1 )
2009 msgItems.emplace_back( _( "Resolved Netclass" ), *netClasses.begin() );
2010 }
2011 }
2012
2013 if( selection.GetSize() >= 2 )
2014 {
2015 bool lengthValid = true;
2016 double selectedLength = 0;
2017
2018 // Lambda to accumulate track length if item is a track or arc, otherwise mark invalid
2019 std::function<void( EDA_ITEM* )> accumulateTrackLength;
2020
2021 accumulateTrackLength =
2022 [&]( EDA_ITEM* aItem )
2023 {
2024 if( aItem->Type() == PCB_TRACE_T || aItem->Type() == PCB_ARC_T )
2025 {
2026 selectedLength += static_cast<PCB_TRACK*>( aItem )->GetLength();
2027 }
2028 else if( aItem->Type() == PCB_VIA_T )
2029 {
2030 // zero 2D length
2031 }
2032 else if( aItem->Type() == PCB_SHAPE_T )
2033 {
2034 PCB_SHAPE* shape = static_cast<PCB_SHAPE*>( aItem );
2035
2036 if( shape->GetShape() == SHAPE_T::SEGMENT
2037 || shape->GetShape() == SHAPE_T::ARC
2038 || shape->GetShape() == SHAPE_T::BEZIER )
2039 {
2040 selectedLength += shape->GetLength();
2041 }
2042 else
2043 {
2044 lengthValid = false;
2045 }
2046 }
2047 // Use dynamic_cast to include PCB_GENERATORs.
2048 else if( PCB_GROUP* group = dynamic_cast<PCB_GROUP*>( aItem ) )
2049 {
2050 group->RunOnChildren( accumulateTrackLength, RECURSE_MODE::NO_RECURSE );
2051 }
2052 else
2053 {
2054 lengthValid = false;
2055 }
2056 };
2057
2058 for( EDA_ITEM* item : selection )
2059 {
2060 if( lengthValid )
2061 accumulateTrackLength( item );
2062 }
2063
2064 if( lengthValid )
2065 {
2066 msgItems.emplace_back( _( "Selected 2D Length" ),
2067 m_frame->MessageTextFromValue( selectedLength ) );
2068 }
2069 }
2070
2071 if( selection.GetSize() >= 2 && selection.GetSize() < 100 )
2072 {
2073 LSET enabledCopper = LSET::AllCuMask( m_frame->GetBoard()->GetCopperLayerCount() );
2074 bool areaValid = true;
2075
2076 std::map<PCB_LAYER_ID, SHAPE_POLY_SET> copperPolys;
2077 SHAPE_POLY_SET holes;
2078
2079 std::function<void( EDA_ITEM* )> accumulateArea;
2080
2081 accumulateArea =
2082 [&]( EDA_ITEM* aItem )
2083 {
2084 if( aItem->Type() == PCB_FOOTPRINT_T )
2085 {
2086 areaValid = false;
2087 return;
2088 }
2089
2090 if( BOARD_ITEM* boardItem = dynamic_cast<BOARD_ITEM*>( aItem ) )
2091 {
2092 boardItem->RunOnChildren( accumulateArea, RECURSE_MODE::NO_RECURSE );
2093
2094 for( PCB_LAYER_ID layer : LSET( boardItem->GetLayerSet() & enabledCopper ) )
2095 {
2096 boardItem->TransformShapeToPolySet( copperPolys[layer], layer, 0,
2098 }
2099
2100 if( aItem->Type() == PCB_PAD_T && static_cast<PAD*>( aItem )->HasHole() )
2101 {
2102 static_cast<PAD*>( aItem )->TransformHoleToPolygon( holes, 0, ARC_LOW_DEF,
2103 ERROR_OUTSIDE );
2104 }
2105 else if( aItem->Type() == PCB_VIA_T )
2106 {
2107 PCB_VIA* via = static_cast<PCB_VIA*>( aItem );
2108 VECTOR2I center = via->GetPosition();
2109 int R = via->GetDrillValue() / 2;
2110
2112 }
2113 }
2114 };
2115
2116 for( EDA_ITEM* item : selection )
2117 {
2118 if( areaValid )
2119 accumulateArea( item );
2120 }
2121
2122 if( areaValid )
2123 {
2124 double area = 0.0;
2125
2126 for( auto& [layer, copperPoly] : copperPolys )
2127 {
2128 copperPoly.BooleanSubtract( holes );
2129 area += copperPoly.Area();
2130 }
2131
2132 msgItems.emplace_back( _( "Selected 2D Copper Area" ),
2133 m_frame->MessageTextFromValue( area, true, EDA_DATA_TYPE::AREA ) );
2134 }
2135 }
2136 }
2137 else
2138 {
2139 m_frame->GetBoard()->GetMsgPanelInfo( m_frame, msgItems );
2140 }
2141
2142 m_frame->SetMsgPanel( msgItems );
2143
2144 return 0;
2145}
2146
2147
2149{
2150 wxFileName fileName = wxFileName( *aEvent.Parameter<wxString*>() );
2151
2152 PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame );
2153
2154 if( !editFrame )
2155 return 1;
2156
2157 wxString filePath = fileName.GetFullPath();
2159 IO_RELEASER<PCB_IO> pi( PCB_IO_MGR::PluginFind( pluginType ) );
2160
2161 if( !pi )
2162 return 1;
2163
2164 return AppendBoard( *pi, filePath );
2165}
2166
2167
2169{
2170 view()->SetMirror( !view()->IsMirroredX(), false );
2171 view()->RecacheAllItems();
2172 frame()->GetCanvas()->ForceRefresh();
2173 frame()->OnDisplayOptionsChanged();
2174 return 0;
2175}
2176
2177// clang-format off
2179{
2182 Go( &PCB_CONTROL::Print, ACTIONS::print.MakeEvent() );
2183 Go( &PCB_CONTROL::Quit, ACTIONS::quit.MakeEvent() );
2184
2185 // Footprint library actions
2190
2191 // Display modes
2207
2208 // Layer control
2246
2249
2250 // Grid control
2253
2254 Go( &PCB_CONTROL::Undo, ACTIONS::undo.MakeEvent() );
2255 Go( &PCB_CONTROL::Redo, ACTIONS::redo.MakeEvent() );
2256
2257 // Snapping control
2262
2263 // Miscellaneous
2265
2266 // Append control
2270
2271 Go( &PCB_CONTROL::Paste, ACTIONS::paste.MakeEvent() );
2273
2280
2281 // Add library by dropping file
2284}
2285// clang-format on
@ ERROR_OUTSIDE
Definition: approximation.h:33
@ ERROR_INSIDE
Definition: approximation.h:34
constexpr int ARC_LOW_DEF
Definition: base_units.h:119
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
static TOOL_ACTION paste
Definition: actions.h:80
static TOOL_ACTION addLibrary
Definition: actions.h:56
static TOOL_ACTION pickerTool
Definition: actions.h:248
static TOOL_ACTION gridResetOrigin
Definition: actions.h:196
static TOOL_ACTION pasteSpecial
Definition: actions.h:81
static TOOL_ACTION highContrastModeCycle
Definition: actions.h:153
static TOOL_ACTION undo
Definition: actions.h:75
static TOOL_ACTION highContrastMode
Definition: actions.h:152
static TOOL_ACTION quit
Definition: actions.h:66
static TOOL_ACTION redo
Definition: actions.h:76
static TOOL_ACTION deleteTool
Definition: actions.h:86
static TOOL_ACTION selectionClear
Clear the current selection.
Definition: actions.h:220
static TOOL_ACTION print
Definition: actions.h:64
static TOOL_ACTION newLibrary
Definition: actions.h:55
static TOOL_ACTION gridSetOrigin
Definition: actions.h:195
static TOOL_ACTION ddAddLibrary
Definition: actions.h:67
static TOOL_ACTION selectItems
Select a list of items (specified as the event parameter)
Definition: actions.h:228
APP_SETTINGS_BASE is a settings class that should be derived for each standalone KiCad application.
Definition: app_settings.h:92
PANEL_DESIGN_BLOCK_CHOOSER m_DesignBlockChooserPanel
Definition: app_settings.h:197
virtual void Push(const wxString &aMessage=wxEmptyString, int aCommitFlags=0) override
Execute the changes.
COMMIT & Stage(EDA_ITEM *aItem, CHANGE_TYPE aChangeType, BASE_SCREEN *aScreen=nullptr) override
Add a change of the item aItem of type aChangeType to the change list.
virtual void Revert() override
Revert the commit by restoring the modified items state.
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
NETINFO_ITEM * GetNet() const
Return #NET_INFO object for a given item.
std::shared_ptr< NET_SETTINGS > m_NetSettings
void SetGridOrigin(const VECTOR2I &aOrigin)
std::shared_ptr< DRC_ENGINE > m_DRCEngine
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 IsOnLayer(PCB_LAYER_ID aLayer) const
Test to see if this object is on the given layer.
Definition: board_item.h:311
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
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 bool IsOnCopperLayer() const
Definition: board_item.h:148
virtual std::shared_ptr< SHAPE_SEGMENT > GetEffectiveHoleShape() const
Definition: board_item.cpp:289
virtual bool HasHole() const
Definition: board_item.h:153
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
LSET GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
Definition: board.cpp:829
NETINFO_ITEM * DpCoupledNet(const NETINFO_ITEM *aNet)
Definition: board.cpp:2082
void SetVisibleLayers(const LSET &aLayerMask)
A proxy function that calls the correspondent function in m_BoardSettings changes the bit-mask of vis...
Definition: board.cpp:861
void MapNets(BOARD *aDestBoard)
Map all nets in the given board to nets with the same name (if any) in the destination board.
Definition: board.cpp:2682
void BuildListOfNets()
Definition: board.h:860
const GENERATORS & Generators() const
Definition: board.h:344
const std::vector< BOARD_CONNECTED_ITEM * > AllConnectedItems()
Definition: board.cpp:2656
const PAGE_INFO & GetPageSettings() const
Definition: board.h:715
void SetProperties(const std::map< wxString, wxString > &aProps)
Definition: board.h:370
const ZONES & Zones() const
Definition: board.h:342
const GROUPS & Groups() const
The groups must maintain the following invariants.
Definition: board.h:365
bool BuildConnectivity(PROGRESS_REPORTER *aReporter=nullptr)
Build or rebuild the board connectivity database for the board, especially the list of connected item...
Definition: board.cpp:185
void SynchronizeNetsAndNetClasses(bool aResetTrackAndViaSizes)
Copy NETCLASS info to each NET, based on NET membership in a NETCLASS.
Definition: board.cpp:2144
FOOTPRINT * GetFirstFootprint() const
Get the first footprint on the board or nullptr.
Definition: board.h:463
TITLE_BLOCK & GetTitleBlock()
Definition: board.h:721
int GetCopperLayerCount() const
Definition: board.cpp:781
const std::map< wxString, wxString > & GetProperties() const
Definition: board.h:369
const FOOTPRINTS & Footprints() const
Definition: board.h:338
void RemoveAll(std::initializer_list< KICAD_T > aTypes={ PCB_NETINFO_T, PCB_MARKER_T, PCB_GROUP_T, PCB_ZONE_T, PCB_GENERATOR_T, PCB_FOOTPRINT_T, PCB_TRACE_T, PCB_SHAPE_T })
An efficient way to remove all items of a certain type from the board.
Definition: board.cpp:1326
const TRACKS & Tracks() const
Definition: board.h:336
void SetPageSettings(const PAGE_INFO &aPageSettings)
Definition: board.h:716
void SetCopperLayerCount(int aCount)
Definition: board.cpp:787
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
void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList) override
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
Definition: board.cpp:1824
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
void SetEnabledLayers(const LSET &aLayerMask)
A proxy function that calls the correspondent function in m_BoardSettings.
Definition: board.cpp:849
void SetTitleBlock(const TITLE_BLOCK &aTitleBlock)
Definition: board.h:723
const DRAWINGS & Drawings() const
Definition: board.h:340
BOARD_ITEM * Parse()
int GetCount() const
Return the number of objects in the list.
Definition: collector.h:82
void Remove(int aIndex)
Remove the item at aIndex (first position is 0).
Definition: collector.h:110
int m_Threshold
Definition: collector.h:235
Color settings are a bit different than most of the settings objects in that there can be more than o...
void SetColor(int aLayer, const COLOR4D &aColor)
COLOR4D GetColor(int aLayer) const
COMMIT & Added(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Notify observers that aItem has been added.
Definition: commit.h:86
COMMIT & Add(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Add a new item to the model.
Definition: commit.h:80
DO_NOT_SHOW_AGAIN m_DoNotShowAgain
LIB_ID GetSelectedLibId(int *aUnit=nullptr) const
DESIGN_BLOCK * GetDesignBlock(const LIB_ID &aLibId, bool aUseCacheLib, bool aShowErrorMsg)
Load design block from design block library table.
const wxString & GetBoardFile() const
Definition: design_block.h:48
int ShowModal() override
wxString GetName() const
Definition: drc_rule.h:160
MINOPTMAX< int > & Value()
Definition: drc_rule.h:153
MINOPTMAX< int > m_Value
Definition: drc_rule.h:190
virtual APP_SETTINGS_BASE * config() const
Return the settings object used in SaveSettings(), and is overloaded in KICAD_MANAGER_FRAME.
bool IsType(FRAME_T aType) const
HOTKEY_CYCLE_POPUP * GetHotkeyPopup()
void DisplayToolMsg(const wxString &msg) override
void SetMsgPanel(const std::vector< MSG_PANEL_ITEM > &aList)
Clear the message panel and populates it with the contents of aList.
virtual COLOR4D GetGridColor()
virtual void Zoom_Automatique(bool aWarpPointer)
Redraw the screen with best zoom level and the best centering that shows all the page or the board.
virtual void CreateHotkeyPopup()
void ForceRefresh()
Force a redraw.
void SetCurrentCursor(KICURSOR aCursor)
Set the current cursor shape for this panel.
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
A set of EDA_ITEMs (i.e., without duplicates).
Definition: eda_group.h:45
virtual bool RemoveItem(EDA_ITEM *aItem)=0
Remove item from group.
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 void SetPosition(const VECTOR2I &aPos)
Definition: eda_item.h:253
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition: eda_item.h:135
virtual wxString GetItemDescription(UNITS_PROVIDER *aUnitsProvider, bool aFull) const
Return a user-visible description string of this item.
Definition: eda_item.cpp:118
const KIID m_Uuid
Definition: eda_item.h:498
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
EDA_GROUP * GetParentGroup() const
Definition: eda_item.h:114
virtual void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList)
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
Definition: eda_item.h:212
virtual void SetParent(EDA_ITEM *aParent)
Definition: eda_item.h:111
SHAPE_T GetShape() const
Definition: eda_shape.h:168
double GetLength() const
Definition: eda_shape.cpp:374
The interactive edit tool.
Definition: edit_tool.h:56
void DeleteItems(const PCB_SELECTION &aItem, bool aIsCut)
Definition: edit_tool.cpp:2473
static const TOOL_EVENT ClearedEvent
Definition: actions.h:338
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 PointSelectedEvent
Definition: actions.h:335
static const TOOL_EVENT ContrastModeChangedByKeyEvent
Definition: actions.h:357
static const TOOL_EVENT ConnectivityChangedEvent
Selected item had a property changed (except movement)
Definition: actions.h:340
static const TOOL_EVENT UnselectedEvent
Definition: actions.h:337
Component library viewer main window.
void AddFootprintToPCB()
Export the current footprint name and close the library browser.
EDA_ANGLE GetOrientation() const
Definition: footprint.h:232
ZONES & Zones()
Definition: footprint.h:217
void SetPath(const KIID_PATH &aPath)
Definition: footprint.h:269
std::deque< PAD * > & Pads()
Definition: footprint.h:211
void ResolveComponentClassNames(BOARD *aBoard, const std::unordered_set< wxString > &aComponentClassNames)
Resolves a set of component class names to this footprint's actual component class.
Definition: footprint.cpp:4058
const std::unordered_set< wxString > & GetTransientComponentClassNames()
Gets the transient component class names.
Definition: footprint.h:1020
void SetReference(const wxString &aReference)
Definition: footprint.h:625
void ClearTransientComponentClassNames()
Remove the transient component class names.
Definition: footprint.h:1026
void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList) override
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
Definition: footprint.cpp:1674
GROUPS & Groups()
Definition: footprint.h:220
void GetFields(std::vector< PCB_FIELD * > &aVector, bool aVisibleOnly) const
Populate a std::vector with PCB_TEXTs.
Definition: footprint.cpp:627
DRAWINGS & GraphicalItems()
Definition: footprint.h:214
A general implementation of a COLLECTORS_GUIDE.
Definition: collectors.h:319
Used when the right click button is pressed, or when the select tool is in effect.
Definition: collectors.h:202
static const std::vector< KICAD_T > BoardLevelItems
A scan list for all primary board items, omitting items which are subordinate to a FOOTPRINT,...
Definition: collectors.h:233
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
Similar to EDA_VIEW_SWITCHER, this dialog is a popup that shows feedback when using a hotkey to cycle...
void Popup(const wxString &aTitle, const wxArrayString &aItems, int aSelection)
virtual void SetProgressReporter(PROGRESS_REPORTER *aReporter)
Set an optional progress reporter.
Definition: io_base.h:89
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
Definition: ki_exception.h:77
virtual const wxString What() const
A composite of Problem() and Where()
Definition: exceptions.cpp:30
Helper class to create more flexible dialogs, including 'do not show again' checkbox handling.
Definition: kidialog.h:43
void DoNotShowCheckbox(wxString file, int line)
Shows the 'do not show again' checkbox.
Definition: kidialog.cpp:51
int ShowModal() override
Definition: kidialog.cpp:95
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
double a
Alpha component.
Definition: color4d.h:395
void SetGridOrigin(const VECTOR2D &aGridOrigin)
Set the origin point for the grid.
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 SetCursorPosition(const VECTOR2D &aPosition, bool aWarpView=true, bool aTriggeredByArrows=false, long aArrowCommand=0)=0
Move cursor to the requested position expressed in world coordinates.
Hold a (potentially large) number of VIEW_ITEMs and renders them on a graphics device provided by the...
Definition: view.h:67
void SetMirror(bool aMirrorX, bool aMirrorY)
Control the mirroring of the VIEW.
Definition: view.cpp:547
virtual void Add(VIEW_ITEM *aItem, int aDrawPriority=-1)
Add a VIEW_ITEM to the view.
Definition: view.cpp:297
virtual void Remove(VIEW_ITEM *aItem)
Remove a VIEW_ITEM from the view.
Definition: view.cpp:332
GAL * GetGAL() const
Return the #GAL this view is using to draw graphical primitives.
Definition: view.h:198
void RecacheAllItems()
Rebuild GAL display lists.
Definition: view.cpp:1439
void UpdateLayerColor(int aLayer)
Apply the new coloring scheme held by RENDER_SETTINGS in case that it has changed.
Definition: view.cpp:744
void MarkDirty()
Force redraw of view on the next rendering.
Definition: view.h:655
Definition: kiid.h:49
All information about a layer pair as stored in the layer pair store.
Management class for layer pairs in a PCB.
Definition: layer_pairs.h:47
std::vector< LAYER_PAIR_INFO > GetEnabledLayerPairs(int &aCurrentIndex) const
Get a vector of all enabled layer pairs, in order.
void SetCurrentLayerPair(const LAYER_PAIR &aPair)
Set the "active" layer pair.
bool IsValid() const
Check if this LID_ID is valid.
Definition: lib_id.h:172
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
LSEQ UIOrder() const
Return the copper, technical and user layers in the order shown in layer widget.
Definition: lset.cpp:712
LSEQ CuStack() const
Return a sequence of copper layers in starting from the front/top and extending to the back/bottom.
Definition: lset.cpp:247
static LSET AllCuMask(int aCuLayerCount=MAX_CU_LAYERS)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Definition: lset.cpp:572
T Min() const
Definition: minoptmax.h:33
bool HasMax() const
Definition: minoptmax.h:38
T Max() const
Definition: minoptmax.h:34
EDA_MSG_PANEL items for displaying messages.
Definition: msgpanel.h:54
Handle the data for a net.
Definition: netinfo.h:56
static NETINFO_ITEM * OrphanedItem()
NETINFO_ITEM meaning that there was no net assigned for an item, as there was no board storing net li...
Definition: netinfo.h:389
Definition: pad.h:54
Describe the page size and margins of a paper page on which to eventually print or plot.
Definition: page_info.h:59
MAGNETIC_SETTINGS m_MagneticItems
static TOOL_ACTION layerToggle
Definition: pcb_actions.h:364
static TOOL_ACTION layerInner12
Definition: pcb_actions.h:340
static TOOL_ACTION nextFootprint
Definition: pcb_actions.h:477
static TOOL_ACTION layerInner8
Definition: pcb_actions.h:336
static TOOL_ACTION zoneDisplayToggle
Definition: pcb_actions.h:323
static TOOL_ACTION previousFootprint
Definition: pcb_actions.h:476
static TOOL_ACTION layerInner3
Definition: pcb_actions.h:331
static TOOL_ACTION layerPrev
Definition: pcb_actions.h:361
static TOOL_ACTION showRatsnest
Definition: pcb_actions.h:312
static TOOL_ACTION zoneFillAll
Definition: pcb_actions.h:388
static TOOL_ACTION layerInner2
Definition: pcb_actions.h:330
static TOOL_ACTION magneticSnapAllLayers
Definition: pcb_actions.h:231
static TOOL_ACTION ddAppendBoard
Drag and drop.
Definition: pcb_actions.h:578
static TOOL_ACTION layerInner25
Definition: pcb_actions.h:353
static TOOL_ACTION magneticSnapActiveLayer
Snapping controls.
Definition: pcb_actions.h:230
static TOOL_ACTION layerAlphaDec
Definition: pcb_actions.h:363
static TOOL_ACTION zoneDisplayFilled
Definition: pcb_actions.h:319
static TOOL_ACTION layerInner24
Definition: pcb_actions.h:352
static TOOL_ACTION viaDisplayMode
Definition: pcb_actions.h:318
static TOOL_ACTION layerInner29
Definition: pcb_actions.h:357
static TOOL_ACTION layerInner11
Definition: pcb_actions.h:339
static TOOL_ACTION layerAlphaInc
Definition: pcb_actions.h:362
static TOOL_ACTION layerPairPresetsCycle
Definition: pcb_actions.h:365
static TOOL_ACTION layerInner16
Definition: pcb_actions.h:344
static TOOL_ACTION layerInner26
Definition: pcb_actions.h:354
static TOOL_ACTION layerInner18
Definition: pcb_actions.h:346
static TOOL_ACTION layerInner14
Definition: pcb_actions.h:342
static TOOL_ACTION trackDisplayMode
Definition: pcb_actions.h:316
static TOOL_ACTION magneticSnapToggle
Definition: pcb_actions.h:232
static TOOL_ACTION layerInner6
Definition: pcb_actions.h:334
static TOOL_ACTION ddImportFootprint
Definition: pcb_actions.h:579
static TOOL_ACTION zoneDisplayTriangulated
Definition: pcb_actions.h:322
static TOOL_ACTION layerInner22
Definition: pcb_actions.h:350
static TOOL_ACTION placeDesignBlock
Definition: pcb_actions.h:440
static TOOL_ACTION layerInner5
Definition: pcb_actions.h:333
static TOOL_ACTION zoneDisplayFractured
Definition: pcb_actions.h:321
static TOOL_ACTION ratsnestModeCycle
Definition: pcb_actions.h:315
static TOOL_ACTION layerInner20
Definition: pcb_actions.h:348
static TOOL_ACTION layerInner7
Definition: pcb_actions.h:335
static TOOL_ACTION layerInner27
Definition: pcb_actions.h:355
static TOOL_ACTION loadFpFromBoard
Definition: pcb_actions.h:473
static TOOL_ACTION appendBoard
Definition: pcb_actions.h:534
static TOOL_ACTION netColorModeCycle
Definition: pcb_actions.h:314
static TOOL_ACTION layerInner1
Definition: pcb_actions.h:329
static TOOL_ACTION layerInner10
Definition: pcb_actions.h:338
static TOOL_ACTION layerInner15
Definition: pcb_actions.h:343
static TOOL_ACTION layerInner17
Definition: pcb_actions.h:345
static TOOL_ACTION flipBoard
Definition: pcb_actions.h:372
static TOOL_ACTION layerBottom
Definition: pcb_actions.h:359
static TOOL_ACTION zoneDisplayOutline
Definition: pcb_actions.h:320
static TOOL_ACTION ratsnestLineMode
Definition: pcb_actions.h:313
static TOOL_ACTION layerInner19
Definition: pcb_actions.h:347
static TOOL_ACTION layerInner9
Definition: pcb_actions.h:337
static TOOL_ACTION move
move or drag an item
Definition: pcb_actions.h:103
static TOOL_ACTION layerInner30
Definition: pcb_actions.h:358
static TOOL_ACTION layerTop
Definition: pcb_actions.h:328
static TOOL_ACTION layerInner4
Definition: pcb_actions.h:332
static TOOL_ACTION layerInner13
Definition: pcb_actions.h:341
static TOOL_ACTION layerInner21
Definition: pcb_actions.h:349
static TOOL_ACTION saveFpToBoard
Definition: pcb_actions.h:474
static TOOL_ACTION layerNext
Definition: pcb_actions.h:360
static TOOL_ACTION layerInner23
Definition: pcb_actions.h:351
static TOOL_ACTION layerInner28
Definition: pcb_actions.h:356
Common, abstract interface for edit frames.
LAYER_PAIR_SETTINGS * GetLayerPairSettings()
Acess to the layer pair settings controller of the board, if available.
bool AddLibrary(const wxString &aLibName=wxEmptyString, FP_LIB_TABLE *aTable=nullptr)
Add an existing library to either the global or project library table.
void RestoreCopyFromUndoList(wxCommandEvent &aEvent)
Undo the last edit:
Definition: undo_redo.cpp:199
APPEARANCE_CONTROLS * GetAppearancePanel()
void RestoreCopyFromRedoList(wxCommandEvent &aEvent)
Redo the last edit:
Definition: undo_redo.cpp:229
Base PCB main window class for Pcbnew, Gerbview, and CvPcb footprint viewer.
const VECTOR2I GetPageSizeIU() const override
Works off of GetPageSettings() to return the size of the paper page in the internal units of this par...
PCBNEW_SETTINGS * GetPcbNewSettings() const
virtual void SwitchLayer(PCB_LAYER_ID aLayer)
Change the active layer in the frame.
virtual PCB_LAYER_ID GetActiveLayer() const
void OnModify() override
Must be called after a change in order to set the "modify" flag and update other data structures and ...
virtual void SaveCopyInUndoList(EDA_ITEM *aItemToCopy, UNDO_REDO aTypeCommand)
Create a new entry in undo list of commands.
PCB_DRAW_PANEL_GAL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
PCB_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
BOARD * GetBoard() const
virtual BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Return the BOARD_DESIGN_SETTINGS for the open project.
void SetDisplayOptions(const PCB_DISPLAY_OPTIONS &aOptions, bool aRefresh=true)
Update the current display options.
GENERAL_COLLECTORS_GUIDE GetCollectorsGuide()
virtual BOARD_ITEM_CONTAINER * GetModel() const =0
FOOTPRINT_EDITOR_SETTINGS * GetFootprintEditorSettings() const
virtual COLOR_SETTINGS * GetColorSettings(bool aForceRefresh=false) const override
Helper to retrieve the current color settings.
void setTransitions() override
< Sets up handlers for various events.
int AppendBoardFromFile(const TOOL_EVENT &aEvent)
int AddLibrary(const TOOL_EVENT &aEvent)
int DdAppendBoard(const TOOL_EVENT &aEvent)
int LoadFpFromBoard(const TOOL_EVENT &aEvent)
int DdImportFootprint(const TOOL_EVENT &aEvent)
int SnapModeFeedback(const TOOL_EVENT &aEvent)
int NetColorModeCycle(const TOOL_EVENT &aEvent)
int SaveFpToBoard(const TOOL_EVENT &aEvent)
int RatsnestModeCycle(const TOOL_EVENT &aEvent)
int TrackDisplayMode(const TOOL_EVENT &aEvent)
int DdAddLibrary(const TOOL_EVENT &aEvent)
int Redo(const TOOL_EVENT &aEvent)
int LayerPresetFeedback(const TOOL_EVENT &aEvent)
int UpdateMessagePanel(const TOOL_EVENT &aEvent)
int LayerAlphaDec(const TOOL_EVENT &aEvent)
int LayerNext(const TOOL_EVENT &aEvent)
std::unique_ptr< STATUS_TEXT_POPUP > m_statusPopup
Definition: pcb_control.h:162
int ToggleRatsnest(const TOOL_EVENT &aEvent)
int AppendBoard(PCB_IO &pi, const wxString &fileName)
int LayerAlphaInc(const TOOL_EVENT &aEvent)
int HighContrastModeCycle(const TOOL_EVENT &aEvent)
std::unique_ptr< KIGFX::ORIGIN_VIEWITEM > m_gridOrigin
Definition: pcb_control.h:158
int Quit(const TOOL_EVENT &aEvent)
int HighContrastMode(const TOOL_EVENT &aEvent)
int Undo(const TOOL_EVENT &aEvent)
int ViaDisplayMode(const TOOL_EVENT &aEvent)
PCB_BASE_FRAME * m_frame
Grid origin marker.
Definition: pcb_control.h:155
static void DoSetGridOrigin(KIGFX::VIEW *aView, PCB_BASE_FRAME *aFrame, EDA_ITEM *originViewItem, const VECTOR2D &aPoint)
void pruneItemLayers(std::vector< BOARD_ITEM * > &aItems)
Helper for pasting.
int GridPlaceOrigin(const TOOL_EVENT &aEvent)
int FlipPcbView(const TOOL_EVENT &aEvent)
int SnapMode(const TOOL_EVENT &aEvent)
bool placeBoardItems(BOARD_COMMIT *aCommit, std::vector< BOARD_ITEM * > &aItems, bool aIsNew, bool aAnchorAtOrigin, bool aReannotateDuplicates)
Add and select or just select for move/place command a list of board items.
int ContrastModeFeedback(const TOOL_EVENT &aEvent)
int LayerToggle(const TOOL_EVENT &aEvent)
void Reset(RESET_REASON aReason) override
Bring the tool to a known, initial state.
int IterateFootprint(const TOOL_EVENT &aEvent)
int Print(const TOOL_EVENT &aEvent)
int ZoneDisplayMode(const TOOL_EVENT &aEvent)
int GridResetOrigin(const TOOL_EVENT &aEvent)
BOARD_ITEM * m_pickerItem
Definition: pcb_control.h:160
int InteractiveDelete(const TOOL_EVENT &aEvent)
int AppendDesignBlock(const TOOL_EVENT &aEvent)
int LayerPrev(const TOOL_EVENT &aEvent)
int CycleLayerPresets(const TOOL_EVENT &aEvent)
int Paste(const TOOL_EVENT &aEvent)
void unfilledZoneCheck()
We have bug reports indicating that some new users confuse zone filling/unfilling with the display mo...
int LayerSwitch(const TOOL_EVENT &aEvent)
Abstract dimension API.
HIGH_CONTRAST_MODE m_ContrastModeDisplay
How inactive layers are displayed.
NET_COLOR_MODE m_NetColorMode
How to use color overrides on specific nets and netclasses.
ZONE_DISPLAY_MODE m_ZoneDisplayMode
void UpdateColors()
Update the color settings in the painter and GAL.
virtual KIGFX::PCB_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
void RedrawRatsnest()
Return the bounding box of the view that should be used if model is not valid.
The main frame for Pcbnew.
PCB_DESIGN_BLOCK_PANE * GetDesignBlockPane() const
static const TOOL_EVENT & SnappingModeChangedByKeyEvent()
Hotkey feedback.
static const TOOL_EVENT & LayerPairPresetChangedByKeyEvent()
A set of BOARD_ITEMs (i.e., without duplicates).
Definition: pcb_group.h:53
bool AddItem(EDA_ITEM *aItem) override
Add item to group.
Definition: pcb_group.cpp:81
static PCB_IO * PluginFind(PCB_FILE_T aFileType)
Return a #PLUGIN which the caller can use to import, export, save, or load design documents.
Definition: pcb_io_mgr.cpp:69
PCB_FILE_T
The set of file types that the PCB_IO_MGR knows about, and for which there has been a plugin written,...
Definition: pcb_io_mgr.h:56
@ KICAD_SEXP
S-expression Pcbnew file format.
Definition: pcb_io_mgr.h:58
static PCB_FILE_T FindPluginTypeFromBoardPath(const wxString &aFileName, int aCtl=0)
Return a plugin type given a path for a board file.
Definition: pcb_io_mgr.cpp:112
A base class that BOARD loading and saving plugins should derive from.
Definition: pcb_io.h:71
virtual void SetQueryUserCallback(std::function< bool(wxString aTitle, int aIcon, wxString aMessage, wxString aAction)> aCallback)
Registers a KIDIALOG callback for collecting info from the user.
Definition: pcb_io.h:97
virtual BOARD * LoadBoard(const wxString &aFileName, BOARD *aAppendToMe, const std::map< std::string, UTF8 > *aProperties=nullptr, PROJECT *aProject=nullptr)
Load information from some input file format that this PCB_IO implementation knows about into either ...
Definition: pcb_io.cpp:74
Class that manages the presentation of PCB layers in a PCB frame.
wxString getLayerPairName(const LAYER_PAIR &aPair) const
Definition: sel_layer.cpp:91
Generic tool for picking an item.
PCB_LAYER_ID m_Route_Layer_TOP
Definition: pcb_screen.h:43
PCB_LAYER_ID m_Route_Layer_BOTTOM
Definition: pcb_screen.h:44
The selection tool: currently supports:
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...
void select(EDA_ITEM *aItem) override
Take necessary action mark an item as selected.
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.
PCB_GROUP * GetEnteredGroup()
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.
int ClearSelection(const TOOL_EVENT &aEvent)
PCB_SELECTION & GetSelection()
EDA_ITEM * GetTopLeftItem(bool aFootprintsOnly=false) const override
T * frame() const
KIGFX::PCB_VIEW * view() const
BOARD * board() const
PCB_DRAW_PANEL_GAL * canvas() const
PCBNEW_SETTINGS::DISPLAY_OPTIONS & displayOptions() const
bool m_isFootprintEditor
const PCB_SELECTION & selection() const
FOOTPRINT * footprint() const
const VECTOR2I & GetStart() const
Definition: pcb_track.h:152
const VECTOR2I & GetEnd() const
Definition: pcb_track.h:149
bool HasValidLayerPair(int aCopperLayerCount)
Definition: pcb_track.cpp:1159
virtual COMMON_SETTINGS * GetCommonSettings() const
Definition: pgm_base.cpp:687
void SetMotionHandler(MOTION_HANDLER aHandler)
Set a handler for mouse motion.
Definition: picker_tool.h:84
void SetClickHandler(CLICK_HANDLER aHandler)
Set a handler for mouse click event.
Definition: picker_tool.h:73
void SetCursor(KICURSOR aCursor)
Definition: picker_tool.h:64
void SetFinalizeHandler(FINALIZE_HANDLER aHandler)
Set a handler for the finalize event.
Definition: picker_tool.h:104
Definition: seg.h:42
ecoord SquaredDistance(const SEG &aSeg) const
Definition: seg.cpp:75
VECTOR2I::extended_type ecoord
Definition: seg.h:44
void BrightenItem(EDA_ITEM *aItem)
void UnbrightenItem(EDA_ITEM *aItem)
virtual void Add(EDA_ITEM *aItem)
Definition: selection.cpp:42
virtual unsigned int GetSize() const override
Return the number of stored items.
Definition: selection.h:100
EDA_ITEM * Front() const
Definition: selection.h:172
int Size() const
Returns the number of selected parts.
Definition: selection.h:116
void SetReferencePoint(const VECTOR2I &aP)
Definition: selection.cpp:178
bool Empty() const
Checks if there is anything selected.
Definition: selection.h:110
Represent a set of closed polygons.
Hold the information shown in the lower right corner of a plot, printout, or editing view.
Definition: title_block.h:41
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
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
@ GAL_SWITCH
Rendering engine changes.
Definition: tool_base.h:82
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).
void Activate()
Run the tool.
bool ProcessEvent(const TOOL_EVENT &aEvent)
Propagate an event to tools that requested events of matching type(s).
void PostEvent(const TOOL_EVENT &aEvent)
Put an event to the event queue to be processed at the end of event processing cycle.
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
bool RunSynchronousAction(const TOOL_ACTION &aAction, COMMIT *aCommit, T aParam)
Run the specified action immediately, pausing the current action to run the new one.
Definition: tool_manager.h:197
wxString MessageTextFromMinOptMax(const MINOPTMAX< int > &aValue) const
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
static constexpr extended_type ECOORD_MAX
Definition: vector2d.h:76
A modified version of the wxInfoBar class that allows us to:
Definition: wx_infobar.h:76
void RemoveAllButtons()
Remove all the buttons that have been added by the user.
Definition: wx_infobar.cpp:353
void ShowMessageFor(const wxString &aMessage, int aTime, int aFlags=wxICON_INFORMATION, MESSAGE_TYPE aType=WX_INFOBAR::MESSAGE_TYPE::GENERIC)
Show the infobar with the provided message and icon for a specific period of time.
Definition: wx_infobar.cpp:140
void AddButton(wxButton *aButton)
Add an already created button to the infobar.
Definition: wx_infobar.cpp:309
Multi-thread safe progress reporter dialog, intended for use of tasks that parallel reporting back of...
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
std::string GetClipboardUTF8()
Return the information currently stored in the system clipboard.
Definition: clipboard.cpp:51
std::unique_ptr< wxImage > GetImageFromClipboard()
Get image data from the clipboard, if there is any.
Definition: clipboard.cpp:77
@ CHT_GROUP
Definition: commit.h:45
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition: confirm.cpp:194
void DisplayError(wxWindow *aParent, const wxString &aText)
Display an error or warning message box with aMessage.
Definition: confirm.cpp:170
This file is part of the common library.
void TransformCircleToPolygon(SHAPE_LINE_CHAIN &aBuffer, const VECTOR2I &aCenter, int aRadius, int aError, ERROR_LOC aErrorLoc, int aMinSegCount=0)
Convert a circle to a polygon, using multiple straight lines.
#define ALPHA_MAX
@ DIFF_PAIR_GAP_CONSTRAINT
Definition: drc_rule.h:70
@ EDGE_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:53
@ CLEARANCE_CONSTRAINT
Definition: drc_rule.h:49
@ MAX_UNCOUPLED_CONSTRAINT
Definition: drc_rule.h:71
@ HOLE_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:51
#define _(s)
Declaration of the eda_3d_viewer class.
std::vector< EDA_ITEM * > EDA_ITEMS
Define list of drawing items for screens.
Definition: eda_item.h:548
#define SKIP_STRUCT
flag indicating that the structure should be ignored
FPVIEWER_CONSTANTS
@ FRAME_PCB_EDITOR
Definition: frame_type.h:42
@ FRAME_FOOTPRINT_VIEWER
Definition: frame_type.h:45
@ FRAME_FOOTPRINT_EDITOR
Definition: frame_type.h:43
wxString KeyNameFromKeyCode(int aKeycode, bool *aIsFound)
Return the key name from the key code.
std::unique_ptr< T > IO_RELEASER
Helper to hold and release an IO_BASE object when exceptions are thrown.
Definition: io_mgr.h:33
This file is part of the common library.
#define KICTL_KICAD_ONLY
chosen file is from KiCad according to user
Definition: kiway_player.h:75
int GetNetnameLayer(int aLayer)
Return a netname layer corresponding to the given layer.
Definition: layer_ids.h:839
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition: layer_ids.h:663
@ LAYER_RATSNEST
Definition: layer_ids.h:252
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ Edge_Cuts
Definition: layer_ids.h:112
@ B_Cu
Definition: layer_ids.h:65
@ Margin
Definition: layer_ids.h:113
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
@ F_Cu
Definition: layer_ids.h:64
#define ZONE_LAYER_FOR(boardLayer)
Definition: layer_ids.h:358
std::optional< wxString > GetMsgPanelDisplayUuid(const KIID &aKiid)
Get a formatted UUID string for display in the message panel, according to the current advanced confi...
Definition: msgpanel.cpp:245
@ REPAINT
Item needs to be redrawn.
Definition: view_item.h:58
bool contains(const _Container &__container, _Value __value)
Returns true if the container contains the given value.
Definition: kicad_algo.h:100
bool AskLoadBoardFileName(PCB_EDIT_FRAME *aParent, wxString *aFileName, int aCtl=0)
Show a wxFileDialog asking for a BOARD filename to open.
#define ALPHA_STEP
static void pasteFootprintItemsToFootprintEditor(FOOTPRINT *aClipFootprint, BOARD *aBoard, std::vector< BOARD_ITEM * > &aPastedItems)
#define ALPHA_MIN
void Flip(T &aValue)
static void moveUnflaggedItems(const std::deque< T > &aList, std::vector< BOARD_ITEM * > &aTarget, bool aIsNew)
Class to handle a set of BOARD_ITEMs.
bool AskLoadBoardFileName(PCB_EDIT_FRAME *aParent, wxString *aFileName, int aCtl=0)
Show a wxFileDialog asking for a BOARD filename to open.
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
#define APPEND_UNDO
Definition: sch_commit.h:42
#define HITTEST_THRESHOLD_PIXELS
std::vector< FAB_LAYER_COLOR > dummy
wxString UnescapeString(const wxString &aSource)
VECTOR2I center
int actual
constexpr KICAD_T BaseType(const KICAD_T aType)
Return the underlying type of the given type.
Definition: typeinfo.h:250
@ PCB_T
Definition: typeinfo.h:82
@ 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
@ 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_TEXT_T
class PCB_TEXT, text on a layer
Definition: typeinfo.h:92
@ 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_DIMENSION_T
class PCB_DIMENSION_BASE: abstract dimension meta-type
Definition: typeinfo.h:100
@ PCB_TABLE_T
class PCB_TABLE, table of PCB_TABLECELLs
Definition: typeinfo.h:94
@ 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
Definition of file extensions used in Kicad.