KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
drawing_tool.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2014-2017 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 "drawing_tool.h"
27#include "geometry/shape_rect.h"
28#include "dialog_table_properties.h"
29
30#include <pgm_base.h>
32#include <pcbnew_settings.h>
34#include <dialogs/dialog_text_properties.h>
45#include <router/router_tool.h>
46#include <status_popup.h>
47#include <tool/tool_manager.h>
48#include <tools/pcb_actions.h>
54#include <view/view.h>
56#include <widgets/wx_infobar.h>
57#include <wx/filedlg.h>
58#include <wx/msgdlg.h>
59
60#include <bitmaps.h>
61#include <board.h>
62#include <board_commit.h>
64#include <confirm.h>
65#include <footprint.h>
66#include <macros.h>
67#include <gal/painter.h>
68#include <pcb_edit_frame.h>
69#include <pcb_group.h>
70#include <pcb_reference_image.h>
71#include <pcb_text.h>
72#include <pcb_textbox.h>
73#include <pcb_table.h>
74#include <pcb_tablecell.h>
75#include <pcb_dimension.h>
76#include <pcbnew_id.h>
77#include <scoped_set_reset.h>
78#include <string_utils.h>
79#include <zone.h>
80#include <fix_board_shape.h>
81#include <view/view_controls.h>
82
83const unsigned int DRAWING_TOOL::COORDS_PADDING = pcbIUScale.mmToIU( 20 );
84
86
87
89{
90public:
92 ACTION_MENU( true )
93 {
94 SetIcon( BITMAPS::width_track_via );
95 SetTitle( _( "Select Via Size" ) );
96 }
97
98protected:
99 ACTION_MENU* create() const override
100 {
101 return new VIA_SIZE_MENU();
102 }
103
104 void update() override
105 {
108 bool useIndex = !bds.m_UseConnectedTrackWidth
109 && !bds.UseCustomTrackViaSize();
110 wxString msg;
111
112 Clear();
113
114 Append( ID_POPUP_PCB_SELECT_CUSTOM_WIDTH, _( "Use Custom Values..." ),
115 _( "Specify custom track and via sizes" ), wxITEM_CHECK );
117
118 AppendSeparator();
119
120 for( unsigned i = 1; i < bds.m_ViasDimensionsList.size(); i++ )
121 {
123
124 if( via.m_Drill > 0 )
125 {
126 msg.Printf( _("Via %s, hole %s" ),
127 frame->MessageTextFromValue( via.m_Diameter ),
128 frame->MessageTextFromValue( via.m_Drill ) );
129 }
130 else
131 {
132 msg.Printf( _( "Via %s" ),
133 frame->MessageTextFromValue( via.m_Diameter ) );
134 }
135
136 int menuIdx = ID_POPUP_PCB_SELECT_VIASIZE1 + i;
137 Append( menuIdx, msg, wxEmptyString, wxITEM_CHECK );
138 Check( menuIdx, useIndex && bds.GetViaSizeIndex() == i );
139 }
140 }
141
142 OPT_TOOL_EVENT eventHandler( const wxMenuEvent& aEvent ) override
143 {
146 int id = aEvent.GetId();
147
148 // On Windows, this handler can be called with an event ID not existing in any
149 // menuitem, so only set flags when we have an ID match.
150
152 {
153 DIALOG_TRACK_VIA_SIZE sizeDlg( frame, bds );
154
155 if( sizeDlg.ShowModal() == wxID_OK )
156 {
157 bds.UseCustomTrackViaSize( true );
158 bds.m_UseConnectedTrackWidth = false;
159 }
160 }
162 {
163 bds.UseCustomTrackViaSize( false );
164 bds.m_UseConnectedTrackWidth = false;
166 }
167
169 }
170};
171
172
174 PCB_TOOL_BASE( "pcbnew.InteractiveDrawing" ),
175 m_view( nullptr ),
176 m_controls( nullptr ),
177 m_board( nullptr ),
178 m_frame( nullptr ),
179 m_mode( MODE::NONE ),
180 m_inDrawingTool( false ),
181 m_layer( UNDEFINED_LAYER ),
182 m_stroke( 1, LINE_STYLE::DEFAULT, COLOR4D::UNSPECIFIED ),
183 m_pickerItem( nullptr ),
184 m_tuningPattern( nullptr )
185{
186}
187
188
190{
191}
192
193
195{
196 auto haveHighlight =
197 [&]( const SELECTION& sel )
198 {
200
201 return !cfg->GetHighlightNetCodes().empty();
202 };
203
204 auto activeToolFunctor =
205 [this]( const SELECTION& aSel )
206 {
207 return m_mode != MODE::NONE;
208 };
209
210 // some interactive drawing tools can undo the last point
211 auto canUndoPoint =
212 [this]( const SELECTION& aSel )
213 {
214 return ( m_mode == MODE::ARC
215 || m_mode == MODE::ZONE
218 || m_mode == MODE::BEZIER
219 || m_mode == MODE::LINE );
220 };
221
222 // functor for tools that can automatically close the outline
223 auto canCloseOutline =
224 [this]( const SELECTION& aSel )
225 {
226 return ( m_mode == MODE::ZONE
229 };
230
231 auto arcToolActive =
232 [this]( const SELECTION& aSel )
233 {
234 return m_mode == MODE::ARC;
235 };
236
237 auto viaToolActive =
238 [this]( const SELECTION& aSel )
239 {
240 return m_mode == MODE::VIA;
241 };
242
243 auto tuningToolActive =
244 [this]( const SELECTION& aSel )
245 {
246 return m_mode == MODE::TUNING;
247 };
248
249 auto dimensionToolActive =
250 [this]( const SELECTION& aSel )
251 {
252 return m_mode == MODE::DIMENSION;
253 };
254
255 CONDITIONAL_MENU& ctxMenu = m_menu->GetMenu();
256
257 // cancel current tool goes in main context menu at the top if present
258 ctxMenu.AddItem( ACTIONS::cancelInteractive, activeToolFunctor, 1 );
259 ctxMenu.AddSeparator( 1 );
260
261 ctxMenu.AddItem( PCB_ACTIONS::clearHighlight, haveHighlight, 2 );
262 ctxMenu.AddSeparator( haveHighlight, 2 );
263
264 // tool-specific actions
265 ctxMenu.AddItem( PCB_ACTIONS::closeOutline, canCloseOutline, 200 );
266 ctxMenu.AddItem( PCB_ACTIONS::deleteLastPoint, canUndoPoint, 200 );
267 ctxMenu.AddItem( PCB_ACTIONS::arcPosture, arcToolActive, 200 );
268 ctxMenu.AddItem( PCB_ACTIONS::spacingIncrease, tuningToolActive, 200 );
269 ctxMenu.AddItem( PCB_ACTIONS::spacingDecrease, tuningToolActive, 200 );
270 ctxMenu.AddItem( PCB_ACTIONS::amplIncrease, tuningToolActive, 200 );
271 ctxMenu.AddItem( PCB_ACTIONS::amplDecrease, tuningToolActive, 200 );
272 ctxMenu.AddItem( PCB_ACTIONS::lengthTunerSettings, tuningToolActive, 200 );
273 ctxMenu.AddItem( PCB_ACTIONS::changeDimensionArrows, dimensionToolActive, 200 );
274
275 ctxMenu.AddCheckItem( PCB_ACTIONS::toggleHV45Mode, !tuningToolActive, 250 );
276 ctxMenu.AddSeparator( 500 );
277
278 std::shared_ptr<VIA_SIZE_MENU> viaSizeMenu = std::make_shared<VIA_SIZE_MENU>();
279 viaSizeMenu->SetTool( this );
280 m_menu->RegisterSubMenu( viaSizeMenu );
281 ctxMenu.AddMenu( viaSizeMenu.get(), viaToolActive, 500 );
282
283 ctxMenu.AddSeparator( 500 );
284
285 // Type-specific sub-menus will be added for us by other tools
286 // For example, zone fill/unfill is provided by the PCB control tool
287
288 // Finally, add the standard zoom/grid items
289 getEditFrame<PCB_BASE_FRAME>()->AddStandardSubMenus( *m_menu.get() );
290
291 return true;
292}
293
294
296{
297 // Init variables used by every drawing tool
298 m_view = getView();
300 m_board = getModel<BOARD>();
301 m_frame = getEditFrame<PCB_BASE_EDIT_FRAME>();
302
303 // Re-initialize session attributes
305
306 if( aReason == RESET_REASON::SHUTDOWN )
307 return;
308
311 m_stroke.SetLineStyle( LINE_STYLE::DEFAULT );
312 m_stroke.SetColor( COLOR4D::UNSPECIFIED );
313
322
324}
325
326
328{
329 return m_mode;
330}
331
332
334{
335 if( m_frame )
336 {
338 bool constrained;
339
341 constrained = mgr.GetAppSettings<PCBNEW_SETTINGS>( "pcbnew" )->m_Use45DegreeLimit;
342 else
343 constrained = mgr.GetAppSettings<FOOTPRINT_EDITOR_SETTINGS>( "fpedit" )->m_Use45Limit;
344
345 m_frame->DisplayConstraintsMsg( constrained ? _( "Constrain to H, V, 45" ) : wxString( "" ) );
346 }
347}
348
349
351{
353 return 0;
354
355 if( m_inDrawingTool )
356 return 0;
357
359
360 BOARD_ITEM* parent = m_frame->GetModel();
361 PCB_SHAPE* line = new PCB_SHAPE( parent );
362 BOARD_COMMIT commit( m_frame );
363 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::LINE );
364 std::optional<VECTOR2D> startingPoint;
365 std::stack<PCB_SHAPE*> committedLines;
366
367 line->SetShape( SHAPE_T::SEGMENT );
368 line->SetFlags( IS_NEW );
369
370 if( aEvent.HasPosition() )
371 startingPoint = getViewControls()->GetCursorPosition( !aEvent.DisableGridSnapping() );
372
373 m_frame->PushTool( aEvent );
374 Activate();
375
376 while( drawShape( aEvent, &line, startingPoint, &committedLines ) )
377 {
378 if( line )
379 {
380 commit.Add( line );
381 commit.Push( _( "Draw Line" ) );
382 startingPoint = VECTOR2D( line->GetEnd() );
383 committedLines.push( line );
384 }
385 else
386 {
387 startingPoint = std::nullopt;
388 }
389
390 line = new PCB_SHAPE( parent );
391 line->SetShape( SHAPE_T::SEGMENT );
392 line->SetFlags( IS_NEW );
393 }
394
395 return 0;
396}
397
398
400{
402 return 0;
403
404 if( m_inDrawingTool )
405 return 0;
406
408
409 bool isTextBox = aEvent.IsAction( &PCB_ACTIONS::drawTextBox );
410 PCB_SHAPE* rect = nullptr;
411 BOARD_COMMIT commit( m_frame );
412 BOARD_ITEM* parent = m_frame->GetModel();
413 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::RECTANGLE );
414 std::optional<VECTOR2D> startingPoint;
415
416 rect = isTextBox ? new PCB_TEXTBOX( parent ) : new PCB_SHAPE( parent );
417 rect->SetShape( SHAPE_T::RECTANGLE );
418 rect->SetFilled( false );
419 rect->SetFlags( IS_NEW );
420
421 if( aEvent.HasPosition() )
422 startingPoint = getViewControls()->GetCursorPosition( !aEvent.DisableGridSnapping() );
423
424 m_frame->PushTool( aEvent );
425 Activate();
426
427 while( drawShape( aEvent, &rect, startingPoint, nullptr ) )
428 {
429 if( rect )
430 {
431 bool cancelled = false;
432
433 if( PCB_TEXTBOX* textbox = dynamic_cast<PCB_TEXTBOX*>( rect ) )
434 cancelled = m_frame->ShowTextBoxPropertiesDialog( textbox ) != wxID_OK;
435
436 if( cancelled )
437 {
438 delete rect;
439 rect = nullptr;
440 }
441 else
442 {
443 rect->Normalize();
444 commit.Add( rect );
445 commit.Push( isTextBox ? _( "Draw Text Box" ) : _( "Draw Rectangle" ) );
446
448 }
449 }
450
451 rect = isTextBox ? new PCB_TEXTBOX( parent ) : new PCB_SHAPE( parent );
452 rect->SetShape( SHAPE_T::RECTANGLE );
453 rect->SetFilled( false );
454 rect->SetFlags( IS_NEW );
455 startingPoint = std::nullopt;
456 }
457
458 return 0;
459}
460
461
463{
465 return 0;
466
467 if( m_inDrawingTool )
468 return 0;
469
471
472 BOARD_ITEM* parent = m_frame->GetModel();
473 PCB_SHAPE* circle = new PCB_SHAPE( parent );
474 BOARD_COMMIT commit( m_frame );
475 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::CIRCLE );
476 std::optional<VECTOR2D> startingPoint;
477
478 circle->SetShape( SHAPE_T::CIRCLE );
479 circle->SetFilled( false );
480 circle->SetFlags( IS_NEW );
481
482 if( aEvent.HasPosition() )
483 startingPoint = getViewControls()->GetCursorPosition( !aEvent.DisableGridSnapping() );
484
485 m_frame->PushTool( aEvent );
486 Activate();
487
488 while( drawShape( aEvent, &circle, startingPoint, nullptr ) )
489 {
490 if( circle )
491 {
492 commit.Add( circle );
493 commit.Push( _( "Draw Circle" ) );
494
496 }
497
498 circle = new PCB_SHAPE( parent );
499 circle->SetShape( SHAPE_T::CIRCLE );
500 circle->SetFilled( false );
501 circle->SetFlags( IS_NEW );
502
503 startingPoint = std::nullopt;
504 }
505
506 return 0;
507}
508
509
511{
513 return 0;
514
515 if( m_inDrawingTool )
516 return 0;
517
519
520 BOARD_ITEM* parent = m_frame->GetModel();
521 PCB_SHAPE* arc = new PCB_SHAPE( parent );
522 BOARD_COMMIT commit( m_frame );
523 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::ARC );
524 std::optional<VECTOR2D> startingPoint;
525
526 arc->SetShape( SHAPE_T::ARC );
527 arc->SetFlags( IS_NEW );
528
529 m_frame->PushTool( aEvent );
530 Activate();
531
532 if( aEvent.HasPosition() )
533 startingPoint = aEvent.Position();
534
535 while( drawArc( aEvent, &arc, startingPoint ) )
536 {
537 if( arc )
538 {
539 commit.Add( arc );
540 commit.Push( _( "Draw Arc" ) );
541
543 }
544
545 arc = new PCB_SHAPE( parent );
546 arc->SetShape( SHAPE_T::ARC );
547 arc->SetFlags( IS_NEW );
548
549 startingPoint = std::nullopt;
550 }
551
552 return 0;
553}
554
555
557{
559 return 0;
560
561 if( m_inDrawingTool )
562 return 0;
563
565
566 BOARD_COMMIT commit( m_frame );
567 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::BEZIER );
568 OPT_VECTOR2I startingPoint, startingC1;
569
570 m_frame->PushTool( aEvent );
571 Activate();
572
573 if( aEvent.HasPosition() )
574 startingPoint = aEvent.Position();
575
577 while( result != DRAW_ONE_RESULT::CANCELLED )
578 {
579 std::unique_ptr<PCB_SHAPE> bezier =
580 drawOneBezier( aEvent, startingPoint, startingC1, result );
581
582 // Anyting other than accepted means no chaining
583 startingPoint = std::nullopt;
584 startingC1 = std::nullopt;
585
586 // If a bezier was created, add it and go again
587 if( bezier )
588 {
589 PCB_SHAPE& bezierRef = *bezier;
590 commit.Add( bezier.release() );
591 commit.Push( _( "Draw Bezier" ) );
592
593 // Don't chain if reset (or accepted and reset)
594 if( result == DRAW_ONE_RESULT::ACCEPTED )
595 {
596 startingPoint = bezierRef.GetEnd();
597
598 // If the last bezier has a zero C2 control arm, allow the user to
599 // define a new C1 control arm for the next one.
600 if( bezierRef.GetEnd() != bezierRef.GetBezierC2() )
601 {
602 // Mirror the control point across the end point to get
603 // a tangent control point
604 startingC1 =
605 bezierRef.GetEnd() - ( bezierRef.GetBezierC2() - bezierRef.GetEnd() );
606 }
607 }
608 }
609 }
610
611 return 0;
612}
613
614
616{
617 if( m_inDrawingTool )
618 return 0;
619
621
623 bool immediateMode = image != nullptr;
625 bool ignorePrimePosition = false;
626 COMMON_SETTINGS* common_settings = Pgm().GetCommonSettings();
627
630 BOARD_COMMIT commit( m_frame );
631
633
634 // Add all the drawable symbols to preview
635 if( image )
636 {
637 image->SetPosition( cursorPos );
639 m_view->AddToPreview( image, false ); // Add, but not give ownership
640 }
641
642 m_frame->PushTool( aEvent );
643
644 auto setCursor =
645 [&]()
646 {
647 if( image )
648 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::MOVING );
649 else
650 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
651 };
652
653 auto cleanup =
654 [&] ()
655 {
659 delete image;
660 image = nullptr;
661 };
662
663 Activate();
664
665 // Must be done after Activate() so that it gets set into the correct context
666 getViewControls()->ShowCursor( true );
667
668 // Set initial cursor
669 setCursor();
670
671 // Prime the pump
672 if( image )
673 {
675 }
676 else if( aEvent.HasPosition() )
677 {
678 m_toolMgr->PrimeTool( aEvent.Position() );
679 }
680 else if( common_settings->m_Input.immediate_actions && !aEvent.IsReactivate() )
681 {
682 m_toolMgr->PrimeTool( { 0, 0 } );
683 ignorePrimePosition = true;
684 }
685
686 // Main loop: keep receiving events
687 while( TOOL_EVENT* evt = Wait() )
688 {
689 setCursor();
690
691 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
692 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
693 cursorPos = GetClampedCoords( grid.BestSnapAnchor( m_controls->GetMousePosition(),
694 { m_frame->GetActiveLayer() },
697 m_controls->ForceCursorPosition( true, cursorPos );
698
699 if( evt->IsCancelInteractive() || ( image && evt->IsAction( &ACTIONS::undo ) ) )
700 {
701 if( image )
702 {
703 cleanup();
704 }
705 else
706 {
707 m_frame->PopTool( aEvent );
708 break;
709 }
710
711 if( immediateMode )
712 {
713 m_frame->PopTool( aEvent );
714 break;
715 }
716 }
717 else if( evt->IsActivate() )
718 {
719 if( image && evt->IsMoveTool() )
720 {
721 // We're already moving our own item; ignore the move tool
722 evt->SetPassEvent( false );
723 continue;
724 }
725
726 if( image )
727 {
728 m_frame->ShowInfoBarMsg( _( "Press <ESC> to cancel image creation." ) );
729 evt->SetPassEvent( false );
730 continue;
731 }
732
733 if( evt->IsMoveTool() )
734 {
735 // Leave ourselves on the stack so we come back after the move
736 break;
737 }
738 else
739 {
740 m_frame->PopTool( aEvent );
741 break;
742 }
743 }
744 else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
745 {
746 if( !image )
747 {
749
750 wxFileDialog dlg( m_frame, _( "Choose Image" ), wxEmptyString, wxEmptyString,
751 _( "Image Files" ) + wxS( " " ) + wxImage::GetImageExtWildcard(),
752 wxFD_OPEN );
753
754 if( dlg.ShowModal() != wxID_OK )
755 continue;
756
757 // If we started with a hotkey which has a position then warp back to that.
758 // Otherwise update to the current mouse position pinned inside the autoscroll
759 // boundaries.
760 if( evt->IsPrime() && !ignorePrimePosition )
761 {
762 cursorPos = grid.Align( evt->Position() );
763 getViewControls()->WarpMouseCursor( cursorPos, true );
764 }
765 else
766 {
768 cursorPos = getViewControls()->GetMousePosition();
769 }
770
771 cursorPos = getViewControls()->GetMousePosition( true );
772
773 wxString fullFilename = dlg.GetPath();
774
775 if( wxFileExists( fullFilename ) )
776 image = new PCB_REFERENCE_IMAGE( m_frame->GetModel(), cursorPos );
777
778 if( !image || !image->GetReferenceImage().ReadImageFile( fullFilename ) )
779 {
780 wxMessageBox( wxString::Format(_( "Could not load image from '%s'." ), fullFilename ) );
781 delete image;
782 image = nullptr;
783 continue;
784 }
785
786 image->SetFlags( IS_NEW | IS_MOVING );
787 image->SetLayer( m_frame->GetActiveLayer() );
788
790 m_view->AddToPreview( image, false ); // Add, but not give ownership
791 m_view->RecacheAllItems(); // Bitmaps are cached in Opengl
792 selectionTool->AddItemToSel( image, false );
793
794 getViewControls()->SetCursorPosition( cursorPos, false );
795 setCursor();
796 m_view->ShowPreview( true );
797 }
798 else
799 {
800 commit.Add( image );
801 commit.Push( _( "Place Image" ) );
802
804
805 image = nullptr;
807
809
810 if( immediateMode )
811 {
812 m_frame->PopTool( aEvent );
813 break;
814 }
815 }
816 }
817 else if( evt->IsClick( BUT_RIGHT ) )
818 {
819 // Warp after context menu only if dragging...
820 if( !image )
822
823 m_menu->ShowContextMenu( selectionTool->GetSelection() );
824 }
825 else if( image && ( evt->IsAction( &ACTIONS::refreshPreview )
826 || evt->IsMotion() ) )
827 {
828 image->SetPosition( cursorPos );
830 m_view->AddToPreview( image, false ); // Add, but not give ownership
831 m_view->RecacheAllItems(); // Bitmaps are cached in Opengl
832 }
833 else if( image && evt->IsAction( &ACTIONS::doDelete ) )
834 {
835 cleanup();
836 }
837 else if( image && ( ZONE_FILLER_TOOL::IsZoneFillAction( evt )
838 || evt->IsAction( &ACTIONS::redo ) ) )
839 {
840 wxBell();
841 }
842 else
843 {
844 evt->SetPassEvent();
845 }
846
847 // Enable autopanning and cursor capture only when there is an image to be placed
848 getViewControls()->SetAutoPan( image != nullptr );
849 getViewControls()->CaptureCursor( image != nullptr );
850 }
851
852 getViewControls()->SetAutoPan( false );
853 getViewControls()->CaptureCursor( false );
854 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
855
856 return 0;
857}
858
859
861{
863 return 0;
864
865 if( m_inDrawingTool )
866 return 0;
867
869
870 COMMON_SETTINGS* common_settings = Pgm().GetCommonSettings();
871 PCB_TEXT* text = nullptr;
872 bool ignorePrimePosition = false;
874 BOARD_COMMIT commit( m_frame );
875 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::TEXT );
877
878 auto setCursor =
879 [&]()
880 {
881 if( text )
882 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::MOVING );
883 else
884 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::TEXT );
885 };
886
887 auto cleanup =
888 [&]()
889 {
892 m_controls->ShowCursor( true );
893 m_controls->SetAutoPan( false );
894 m_controls->CaptureCursor( false );
895 delete text;
896 text = nullptr;
897 };
898
900
901 m_frame->PushTool( aEvent );
902
903 Activate();
904 // Must be done after Activate() so that it gets set into the correct context
905 m_controls->ShowCursor( true );
907 // do not capture or auto-pan until we start placing some text
908 // Set initial cursor
909 setCursor();
910
911 if( aEvent.HasPosition() )
912 {
913 m_toolMgr->PrimeTool( aEvent.Position() );
914 }
915 else if( common_settings->m_Input.immediate_actions && !aEvent.IsReactivate() )
916 {
917 m_toolMgr->PrimeTool( { 0, 0 } );
918 ignorePrimePosition = true;
919 }
920
921 // Main loop: keep receiving events
922 while( TOOL_EVENT* evt = Wait() )
923 {
924 setCursor();
925
926 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
927 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
928 VECTOR2I cursorPos =
930 { m_frame->GetActiveLayer() }, GRID_TEXT ),
932 m_controls->ForceCursorPosition( true, cursorPos );
933
934 if( evt->IsDrag() )
935 {
936 continue;
937 }
938 else if( evt->IsCancelInteractive() || ( text && evt->IsAction( &ACTIONS::undo ) ) )
939 {
940 if( text )
941 {
942 cleanup();
943 }
944 else
945 {
946 m_frame->PopTool( aEvent );
947 break;
948 }
949 }
950 else if( evt->IsActivate() )
951 {
952 if( text )
953 cleanup();
954
955 if( evt->IsMoveTool() )
956 {
957 // leave ourselves on the stack so we come back after the move
958 break;
959 }
960 else
961 {
962 m_frame->PopTool( aEvent );
963 break;
964 }
965 }
966 else if( evt->IsClick( BUT_RIGHT ) )
967 {
968 if( !text )
970
971 m_menu->ShowContextMenu( selection() );
972 }
973 else if( evt->IsClick( BUT_LEFT ) )
974 {
975 bool placing = text != nullptr;
976
977 if( !text )
978 {
980
982
984 TEXT_ATTRIBUTES textAttrs;
985
986 textAttrs.m_Size = bds.GetTextSize( layer );
987 textAttrs.m_StrokeWidth = bds.GetTextThickness( layer );
988 InferBold( &textAttrs );
989 textAttrs.m_Italic = bds.GetTextItalic( layer );
990 textAttrs.m_KeepUpright = bds.GetTextUpright( layer );
991 textAttrs.m_Mirrored = IsBackLayer( layer );
992 textAttrs.m_Halign = GR_TEXT_H_ALIGN_LEFT;
994
996 text = new PCB_TEXT( static_cast<FOOTPRINT*>( m_frame->GetModel() ) );
997 else
998 text = new PCB_TEXT( m_frame->GetModel() );
999
1000 text->SetLayer( layer );
1001 text->SetAttributes( textAttrs );
1002 text->SetTextPos( cursorPos );
1003 text->SetFlags( IS_NEW ); // Prevent double undo commits
1004
1005 DIALOG_TEXT_PROPERTIES textDialog( m_frame, text );
1006 bool cancelled;
1007
1009 [&]()
1010 {
1011 // QuasiModal required for Scintilla auto-complete
1012 cancelled = !textDialog.ShowQuasiModal();
1013 } );
1014
1015 if( cancelled || NoPrintableChars( text->GetText() ) )
1016 {
1017 delete text;
1018 text = nullptr;
1019 }
1020 else if( text->GetTextPos() != cursorPos )
1021 {
1022 // If the user modified the location then go ahead and place it there.
1023 // Otherwise we'll drag.
1024 placing = true;
1025 }
1026
1027 if( text )
1028 {
1029 if( !m_view->IsLayerVisible( text->GetLayer() ) )
1030 {
1031 m_frame->GetAppearancePanel()->SetLayerVisible( text->GetLayer(), true );
1033 }
1034
1036 m_view->Update( &selection() );
1037
1038 // update the cursor so it looks correct before another event
1039 setCursor();
1040 }
1041 }
1042
1043 if( placing )
1044 {
1045 text->ClearFlags();
1047
1048 commit.Add( text );
1049 commit.Push( _( "Draw Text" ) );
1050
1052
1053 text = nullptr;
1054 }
1055
1057
1058 // If we started with a hotkey which has a position then warp back to that.
1059 // Otherwise update to the current mouse position pinned inside the autoscroll
1060 // boundaries.
1061 if( evt->IsPrime() && !ignorePrimePosition )
1062 {
1063 cursorPos = evt->Position();
1064 m_controls->WarpMouseCursor( cursorPos, true );
1065 }
1066 else
1067 {
1069 cursorPos = m_controls->GetMousePosition();
1070 }
1071
1073
1074 m_controls->ShowCursor( true );
1075 m_controls->CaptureCursor( text != nullptr );
1076 m_controls->SetAutoPan( text != nullptr );
1077 }
1078 else if( text && ( evt->IsMotion() || evt->IsAction( &PCB_ACTIONS::refreshPreview ) ) )
1079 {
1080 text->SetPosition( cursorPos );
1081 selection().SetReferencePoint( cursorPos );
1082 m_view->Update( &selection() );
1083 }
1084 else if( text
1086 || evt->IsAction( &ACTIONS::redo ) ) )
1087 {
1088 wxBell();
1089 }
1090 else if( text && evt->IsAction( &PCB_ACTIONS::properties ) )
1091 {
1092 frame()->OnEditItemRequest( text );
1093 m_view->Update( &selection() );
1094 frame()->SetMsgPanel( text );
1095 }
1096 else
1097 {
1098 evt->SetPassEvent();
1099 }
1100 }
1101
1102 m_controls->SetAutoPan( false );
1103 m_controls->CaptureCursor( false );
1105 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
1106
1107 if( selection().Empty() )
1109
1110 return 0;
1111}
1112
1113
1115{
1116 if( m_inDrawingTool )
1117 return 0;
1118
1120
1121 PCB_TABLE* table = nullptr;
1123 BOARD_COMMIT commit( m_frame );
1125
1126 // We might be running as the same shape in another co-routine. Make sure that one
1127 // gets whacked.
1129
1130 auto setCursor =
1131 [&]()
1132 {
1133 if( table )
1134 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::MOVING );
1135 else
1136 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::PENCIL );
1137 };
1138
1139 auto cleanup =
1140 [&] ()
1141 {
1144 m_controls->ShowCursor( true );
1145 m_controls->SetAutoPan( false );
1146 m_controls->CaptureCursor( false );
1147 delete table;
1148 table = nullptr;
1149 };
1150
1152
1153 m_frame->PushTool( aEvent );
1154
1155 Activate();
1156 // Must be done after Activate() so that it gets set into the correct context
1157 getViewControls()->ShowCursor( true );
1159 // Set initial cursor
1160 setCursor();
1161
1162 if( aEvent.HasPosition() )
1163 m_toolMgr->PrimeTool( aEvent.Position() );
1164
1165 // Main loop: keep receiving events
1166 while( TOOL_EVENT* evt = Wait() )
1167 {
1168 setCursor();
1169 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
1170 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
1171 VECTOR2I cursorPos =
1173 { m_frame->GetActiveLayer() }, GRID_TEXT ),
1175 m_controls->ForceCursorPosition( true, cursorPos );
1176
1177 if( evt->IsCancelInteractive() || ( table && evt->IsAction( &ACTIONS::undo ) )
1178 || evt->IsDrag() )
1179 {
1180 if( table )
1181 {
1182 cleanup();
1183 }
1184 else
1185 {
1186 m_frame->PopTool( aEvent );
1187 break;
1188 }
1189 }
1190 else if( evt->IsActivate() )
1191 {
1192 if( table )
1193 cleanup();
1194
1195 if( evt->IsMoveTool() )
1196 {
1197 // leave ourselves on the stack so we come back after the move
1198 break;
1199 }
1200 else
1201 {
1202 m_frame->PopTool( aEvent );
1203 break;
1204 }
1205 }
1206 else if( evt->IsClick( BUT_RIGHT ) )
1207 {
1208 // Warp after context menu only if dragging...
1209 if( !table )
1211
1212 m_menu->ShowContextMenu( selection() );
1213 }
1214 else if( evt->IsClick( BUT_LEFT ) )
1215 {
1216 if( !table )
1217 {
1219
1221
1222 table = new PCB_TABLE( m_frame->GetModel(), bds.GetLineThickness( layer ) );
1223 table->SetFlags( IS_NEW );
1224 table->SetLayer( layer );
1225 table->SetColCount( 1 );
1226 table->AddCell( new PCB_TABLECELL( table ) );
1227
1228 table->SetLayer( layer );
1229 table->SetPosition( cursorPos );
1230
1231 if( !m_view->IsLayerVisible( layer ) )
1232 {
1233 m_frame->GetAppearancePanel()->SetLayerVisible( layer, true );
1235 }
1236
1238 m_view->Update( &selection() );
1239
1240 // update the cursor so it looks correct before another event
1241 setCursor();
1242 }
1243 else
1244 {
1246
1247 table->Normalize();
1248
1250
1251 // QuasiModal required for Scintilla auto-complete
1252 if( dlg.ShowQuasiModal() == wxID_OK )
1253 {
1254 commit.Add( table, m_frame->GetScreen() );
1255 commit.Push( _( "Draw Table" ) );
1256
1259 }
1260 else
1261 {
1262 delete table;
1263 }
1264
1265 table = nullptr;
1266 }
1267 }
1268 else if( table && ( evt->IsAction( &ACTIONS::refreshPreview ) || evt->IsMotion() ) )
1269 {
1270 VECTOR2I fontSize = bds.GetTextSize( table->GetLayer() );
1271 VECTOR2I gridSize = grid.GetGridSize( grid.GetItemGrid( table ) );
1272 VECTOR2I origin( table->GetPosition() );
1273 VECTOR2I requestedSize( cursorPos - origin );
1274
1275 int colCount = std::max( 1, requestedSize.x / ( fontSize.x * 15 ) );
1276 int rowCount = std::max( 1, requestedSize.y / ( fontSize.y * 3 ) );
1277
1278 VECTOR2I cellSize( std::max( fontSize.x * 5, requestedSize.x / colCount ),
1279 std::max( fontSize.y * 3, requestedSize.y / rowCount ) );
1280
1281 cellSize.x = KiROUND( (double) cellSize.x / gridSize.x ) * gridSize.x;
1282 cellSize.y = KiROUND( (double) cellSize.y / gridSize.y ) * gridSize.y;
1283
1284 table->ClearCells();
1285 table->SetColCount( colCount );
1286
1287 for( int col = 0; col < colCount; ++col )
1288 table->SetColWidth( col, cellSize.x );
1289
1290 for( int row = 0; row < rowCount; ++row )
1291 {
1292 table->SetRowHeight( row, cellSize.y );
1293
1294 for( int col = 0; col < colCount; ++col )
1295 {
1296 PCB_TABLECELL* cell = new PCB_TABLECELL( table );
1297 cell->SetPosition( origin + VECTOR2I( col * cellSize.x, row * cellSize.y ) );
1298 cell->SetEnd( cell->GetPosition() + cellSize );
1299 table->AddCell( cell );
1300 }
1301 }
1302
1303 selection().SetReferencePoint( cursorPos );
1304 m_view->Update( &selection() );
1306 }
1307 else if( table && evt->IsAction( &PCB_ACTIONS::properties ) )
1308 {
1309 frame()->OnEditItemRequest( table );
1310 m_view->Update( &selection() );
1311 frame()->SetMsgPanel( table );
1312 }
1313 else if( table && ( ZONE_FILLER_TOOL::IsZoneFillAction( evt )
1314 || evt->IsAction( &ACTIONS::redo ) ) )
1315 {
1316 wxBell();
1317 }
1318 else
1319 {
1320 evt->SetPassEvent();
1321 }
1322
1323 // Enable autopanning and cursor capture only when there is a shape being drawn
1324 getViewControls()->SetAutoPan( table != nullptr );
1325 getViewControls()->CaptureCursor( table != nullptr );
1326 }
1327
1328 getViewControls()->SetAutoPan( false );
1329 getViewControls()->CaptureCursor( false );
1330 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
1331 return 0;
1332}
1333
1334
1336{
1337 const VECTOR2I lineVector{ aDim->GetEnd() - aDim->GetStart() };
1338
1339 aDim->SetEnd( aDim->GetStart() + GetVectorSnapped45( lineVector ) );
1340 aDim->Update();
1341}
1342
1343
1345{
1347 return 0;
1348
1349 if( m_inDrawingTool )
1350 return 0;
1351
1353
1354 enum DIMENSION_STEPS
1355 {
1356 SET_ORIGIN = 0,
1357 SET_END,
1358 SET_HEIGHT,
1359 FINISHED
1360 };
1361
1362 TOOL_EVENT originalEvent = aEvent;
1363 PCB_DIMENSION_BASE* dimension = nullptr;
1364 BOARD_COMMIT commit( m_frame );
1367 PCB_SELECTION preview; // A VIEW_GROUP that serves as a preview for the new item(s)
1368 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::DIMENSION );
1369 int step = SET_ORIGIN;
1371
1372 m_view->Add( &preview );
1373
1374 auto cleanup =
1375 [&]()
1376 {
1377 m_controls->SetAutoPan( false );
1378 m_controls->CaptureCursor( false );
1380
1381 preview.Clear();
1382 m_view->Update( &preview );
1383
1384 delete dimension;
1385 dimension = nullptr;
1386 step = SET_ORIGIN;
1387 };
1388
1389 auto setCursor =
1390 [&]()
1391 {
1392 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::MEASURE );
1393 };
1394
1396
1397 m_frame->PushTool( aEvent );
1398
1399 Activate();
1400 // Must be done after Activate() so that it gets set into the correct context
1401 m_controls->ShowCursor( true );
1403 // Set initial cursor
1404 setCursor();
1405
1407
1408 if( aEvent.HasPosition() )
1409 m_toolMgr->PrimeTool( aEvent.Position() );
1410
1411 // Main loop: keep receiving events
1412 while( TOOL_EVENT* evt = Wait() )
1413 {
1414 if( step > SET_ORIGIN )
1415 frame()->SetMsgPanel( dimension );
1416
1417 setCursor();
1418
1419 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
1420 bool is45Limited = Is45Limited() && !evt->Modifier( MD_CTRL );
1421 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
1422
1423 if( step == SET_HEIGHT && t != PCB_DIM_ORTHOGONAL_T )
1424 {
1425 if( dimension->GetStart().x != dimension->GetEnd().x
1426 && dimension->GetStart().y != dimension->GetEnd().y )
1427 {
1428 // Not cardinal. Grid snapping doesn't make sense for height.
1429 grid.SetUseGrid( false );
1430 }
1431 }
1432
1433 VECTOR2I cursorPos = evt->HasPosition() ? evt->Position() : m_controls->GetMousePosition();
1434 cursorPos = GetClampedCoords( grid.BestSnapAnchor( cursorPos, nullptr, GRID_GRAPHICS ),
1436
1437 m_controls->ForceCursorPosition( true, cursorPos );
1438
1439 if( evt->IsCancelInteractive() || ( dimension && evt->IsAction( &ACTIONS::undo ) ) )
1440 {
1441 m_controls->SetAutoPan( false );
1442
1443 if( step != SET_ORIGIN ) // start from the beginning
1444 {
1445 cleanup();
1446 }
1447 else
1448 {
1449 m_frame->PopTool( aEvent );
1450 break;
1451 }
1452 }
1453 else if( evt->IsActivate() )
1454 {
1455 if( step != SET_ORIGIN )
1456 cleanup();
1457
1458 if( evt->IsPointEditor() )
1459 {
1460 // don't exit (the point editor runs in the background)
1461 }
1462 else if( evt->IsMoveTool() )
1463 {
1464 // leave ourselves on the stack so we come back after the move
1465 break;
1466 }
1467 else
1468 {
1469 m_frame->PopTool( aEvent );
1470 break;
1471 }
1472 }
1473 else if( evt->IsAction( &PCB_ACTIONS::incWidth ) && step != SET_ORIGIN )
1474 {
1476 dimension->SetLineThickness( m_stroke.GetWidth() );
1477 m_view->Update( &preview );
1478 frame()->SetMsgPanel( dimension );
1479 }
1480 else if( evt->IsAction( &PCB_ACTIONS::decWidth ) && step != SET_ORIGIN )
1481 {
1482 if( (unsigned) m_stroke.GetWidth() > WIDTH_STEP )
1483 {
1485 dimension->SetLineThickness( m_stroke.GetWidth() );
1486 m_view->Update( &preview );
1487 frame()->SetMsgPanel( dimension );
1488 }
1489 }
1490 else if( evt->IsClick( BUT_RIGHT ) )
1491 {
1492 if( !dimension )
1494
1495 m_menu->ShowContextMenu( selection() );
1496 }
1497 else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
1498 {
1499 switch( step )
1500 {
1501 case SET_ORIGIN:
1502 {
1504
1506
1507 // Init the new item attributes
1508 auto setMeasurementAttributes =
1509 [&]( PCB_DIMENSION_BASE* aDim )
1510 {
1511 aDim->SetUnitsMode( boardSettings.m_DimensionUnitsMode );
1512 aDim->SetUnitsFormat( boardSettings.m_DimensionUnitsFormat );
1513 aDim->SetPrecision( boardSettings.m_DimensionPrecision );
1514 aDim->SetSuppressZeroes( boardSettings.m_DimensionSuppressZeroes );
1515 aDim->SetTextPositionMode( boardSettings.m_DimensionTextPosition );
1516 aDim->SetKeepTextAligned( boardSettings.m_DimensionKeepTextAligned );
1517 };
1518
1519 if( originalEvent.IsAction( &PCB_ACTIONS::drawAlignedDimension ) )
1520 {
1521 dimension = new PCB_DIM_ALIGNED( m_frame->GetModel() );
1522 setMeasurementAttributes( dimension );
1523 }
1524 else if( originalEvent.IsAction( &PCB_ACTIONS::drawOrthogonalDimension ) )
1525 {
1526 dimension = new PCB_DIM_ORTHOGONAL( m_frame->GetModel() );
1527 setMeasurementAttributes( dimension );
1528 }
1529 else if( originalEvent.IsAction( &PCB_ACTIONS::drawCenterDimension ) )
1530 {
1531 dimension = new PCB_DIM_CENTER( m_frame->GetModel() );
1532 }
1533 else if( originalEvent.IsAction( &PCB_ACTIONS::drawRadialDimension ) )
1534 {
1535 dimension = new PCB_DIM_RADIAL( m_frame->GetModel() );
1536 setMeasurementAttributes( dimension );
1537 }
1538 else if( originalEvent.IsAction( &PCB_ACTIONS::drawLeader ) )
1539 {
1540 dimension = new PCB_DIM_LEADER( m_frame->GetModel() );
1541 dimension->SetTextPos( cursorPos );
1542 }
1543 else
1544 {
1545 wxFAIL_MSG( wxT( "Unhandled action in DRAWING_TOOL::DrawDimension" ) );
1546 }
1547
1548 t = dimension->Type();
1549
1550 dimension->SetLayer( layer );
1551 dimension->SetMirrored( IsBackLayer( layer ) );
1552 dimension->SetTextSize( boardSettings.GetTextSize( layer ) );
1553 dimension->SetTextThickness( boardSettings.GetTextThickness( layer ) );
1554 dimension->SetItalic( boardSettings.GetTextItalic( layer ) );
1555 dimension->SetLineThickness( boardSettings.GetLineThickness( layer ) );
1556 dimension->SetArrowLength( boardSettings.m_DimensionArrowLength );
1557 dimension->SetExtensionOffset( boardSettings.m_DimensionExtensionOffset );
1558 dimension->SetStart( cursorPos );
1559 dimension->SetEnd( cursorPos );
1560 dimension->Update();
1561
1562 if( !m_view->IsLayerVisible( layer ) )
1563 {
1564 m_frame->GetAppearancePanel()->SetLayerVisible( layer, true );
1566 }
1567
1568 preview.Add( dimension );
1569 frame()->SetMsgPanel( dimension );
1570
1571 m_controls->SetAutoPan( true );
1572 m_controls->CaptureCursor( true );
1573 break;
1574 }
1575
1576 case SET_END:
1577 // Dimensions that have origin and end in the same spot are not valid
1578 if( dimension->GetStart() == dimension->GetEnd() )
1579 {
1580 --step;
1581 break;
1582 }
1583
1584 if( t == PCB_DIM_CENTER_T || t == PCB_DIM_RADIAL_T || t == PCB_DIM_LEADER_T )
1585 {
1586 // No separate height step
1587 ++step;
1589 }
1590 else
1591 {
1592 break;
1593 }
1594
1595 case SET_HEIGHT:
1596 assert( dimension->GetStart() != dimension->GetEnd() );
1597 assert( dimension->GetLineThickness() > 0 );
1598
1599 preview.Remove( dimension );
1600
1601 commit.Add( dimension );
1602 commit.Push( _( "Draw Dimension" ) );
1603
1604 // Run the edit immediately to set the leader text
1605 if( t == PCB_DIM_LEADER_T )
1606 frame()->OnEditItemRequest( dimension );
1607
1609
1610 break;
1611 }
1612
1613 if( ++step >= FINISHED )
1614 {
1615 dimension = nullptr;
1616 step = SET_ORIGIN;
1617 m_controls->SetAutoPan( false );
1618 m_controls->CaptureCursor( false );
1619 }
1620 else if( evt->IsDblClick( BUT_LEFT ) )
1621 {
1623 }
1624 }
1625 else if( evt->IsMotion() )
1626 {
1627 switch( step )
1628 {
1629 case SET_END:
1630 dimension->SetEnd( cursorPos );
1631
1632 if( is45Limited || t == PCB_DIM_CENTER_T )
1633 constrainDimension( dimension );
1634
1635 if( t == PCB_DIM_ORTHOGONAL_T )
1636 {
1637 PCB_DIM_ORTHOGONAL* ortho = static_cast<PCB_DIM_ORTHOGONAL*>( dimension );
1638
1639 BOX2I bounds( dimension->GetStart(),
1640 dimension->GetEnd() - dimension->GetStart() );
1641
1642 // Create a nice preview by measuring the longer dimension
1643 bool vert = bounds.GetWidth() < bounds.GetHeight();
1644
1645 ortho->SetOrientation( vert ? PCB_DIM_ORTHOGONAL::DIR::VERTICAL
1647 }
1648 else if( t == PCB_DIM_RADIAL_T )
1649 {
1650 PCB_DIM_RADIAL* radialDim = static_cast<PCB_DIM_RADIAL*>( dimension );
1651 VECTOR2I textOffset( radialDim->GetArrowLength() * 10, 0 );
1652
1653 if( radialDim->GetEnd().x < radialDim->GetStart().x )
1654 textOffset = -textOffset;
1655
1656 radialDim->SetTextPos( radialDim->GetKnee() + textOffset );
1657 }
1658 else if( t == PCB_DIM_LEADER_T )
1659 {
1660 VECTOR2I textOffset( dimension->GetArrowLength() * 10, 0 );
1661
1662 if( dimension->GetEnd().x < dimension->GetStart().x )
1663 textOffset = -textOffset;
1664
1665 dimension->SetTextPos( dimension->GetEnd() + textOffset );
1666 }
1667
1668 dimension->Update();
1669 break;
1670
1671 case SET_HEIGHT:
1672 if( t == PCB_DIM_ALIGNED_T )
1673 {
1674 PCB_DIM_ALIGNED* aligned = static_cast<PCB_DIM_ALIGNED*>( dimension );
1675
1676 // Calculating the direction of travel perpendicular to the selected axis
1677 double angle = aligned->GetAngle() + ( M_PI / 2 );
1678
1679 VECTOR2I delta( (VECTOR2I) cursorPos - dimension->GetEnd() );
1680 double height = ( delta.x * cos( angle ) ) + ( delta.y * sin( angle ) );
1681 aligned->SetHeight( height );
1682 aligned->Update();
1683 }
1684 else if( t == PCB_DIM_ORTHOGONAL_T )
1685 {
1686 PCB_DIM_ORTHOGONAL* ortho = static_cast<PCB_DIM_ORTHOGONAL*>( dimension );
1687
1688 BOX2I bbox( dimension->GetStart(),
1689 dimension->GetEnd() - dimension->GetStart() );
1690 VECTOR2I direction( cursorPos - bbox.Centre() );
1691 bool vert;
1692
1693 // Only change the orientation when we move outside the bbox
1694 if( !bbox.Contains( cursorPos ) )
1695 {
1696 // If the dimension is horizontal or vertical, set correct orientation
1697 // otherwise, test if we're left/right of the bounding box or above/below it
1698 if( bbox.GetWidth() == 0 )
1699 vert = true;
1700 else if( bbox.GetHeight() == 0 )
1701 vert = false;
1702 else if( cursorPos.x > bbox.GetLeft() && cursorPos.x < bbox.GetRight() )
1703 vert = false;
1704 else if( cursorPos.y > bbox.GetTop() && cursorPos.y < bbox.GetBottom() )
1705 vert = true;
1706 else
1707 vert = std::abs( direction.y ) < std::abs( direction.x );
1708
1709 ortho->SetOrientation( vert ? PCB_DIM_ORTHOGONAL::DIR::VERTICAL
1711 }
1712 else
1713 {
1714 vert = ortho->GetOrientation() == PCB_DIM_ORTHOGONAL::DIR::VERTICAL;
1715 }
1716
1717 VECTOR2I heightVector( cursorPos - dimension->GetStart() );
1718 ortho->SetHeight( vert ? heightVector.x : heightVector.y );
1719 ortho->Update();
1720 }
1721
1722 break;
1723 }
1724
1725 // Show a preview of the item
1726 m_view->Update( &preview );
1727 }
1728 else if( dimension && evt->IsAction( &PCB_ACTIONS::layerChanged ) )
1729 {
1731
1732 if( !m_view->IsLayerVisible( layer ) )
1733 {
1734 m_frame->GetAppearancePanel()->SetLayerVisible( layer, true );
1736 }
1737
1738 dimension->SetLayer( layer );
1739 dimension->SetTextSize( boardSettings.GetTextSize( layer ) );
1740 dimension->SetTextThickness( boardSettings.GetTextThickness( layer ) );
1741 dimension->SetItalic( boardSettings.GetTextItalic( layer ) );
1742 dimension->SetLineThickness( boardSettings.GetLineThickness( layer ) );
1743 dimension->Update();
1744
1745 m_view->Update( &preview );
1746 frame()->SetMsgPanel( dimension );
1747 }
1748 else if( dimension && evt->IsAction( &PCB_ACTIONS::properties ) )
1749 {
1750 if( step == SET_END || step == SET_HEIGHT )
1751 {
1752 frame()->OnEditItemRequest( dimension );
1753 dimension->Update();
1754 frame()->SetMsgPanel( dimension );
1755 break;
1756 }
1757 else
1758 {
1759 wxBell();
1760 }
1761 }
1762 else if( dimension && evt->IsAction( &PCB_ACTIONS::changeDimensionArrows ) )
1763 {
1764 switch( dimension->Type() )
1765 {
1766 case PCB_DIM_ALIGNED_T:
1768 case PCB_DIM_RADIAL_T:
1769 if( dimension->GetArrowDirection() == DIM_ARROW_DIRECTION::INWARD )
1770 dimension->SetArrowDirection( DIM_ARROW_DIRECTION::OUTWARD );
1771 else
1772 dimension->SetArrowDirection( DIM_ARROW_DIRECTION::INWARD );
1773 break;
1774 default:
1775 // Other dimension types don't have arrows that can swap
1776 wxBell();
1777 }
1778
1779 m_view->Update( &preview );
1780 }
1781 else if( dimension && ( ZONE_FILLER_TOOL::IsZoneFillAction( evt )
1782 || evt->IsAction( &ACTIONS::redo ) ) )
1783 {
1784 wxBell();
1785 }
1786 else
1787 {
1788 evt->SetPassEvent();
1789 }
1790 }
1791
1792 if( step != SET_ORIGIN )
1793 delete dimension;
1794
1795 m_controls->SetAutoPan( false );
1797 m_controls->CaptureCursor( false );
1798 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
1799
1800 m_view->Remove( &preview );
1801
1802 if( selection().Empty() )
1804
1805 return 0;
1806}
1807
1808
1810{
1811 if( !m_frame->GetModel() )
1812 return 0;
1813
1814 if( m_inDrawingTool )
1815 return 0;
1816
1818
1820 int dlgResult = dlg.ShowModal();
1821
1822 std::list<std::unique_ptr<EDA_ITEM>>& list = dlg.GetImportedItems();
1823
1824 if( dlgResult != wxID_OK )
1825 return 0;
1826
1827 // Ensure the list is not empty:
1828 if( list.empty() )
1829 {
1830 wxMessageBox( _( "No graphic items found in file.") );
1831 return 0;
1832 }
1833
1835
1836 std::vector<BOARD_ITEM*> newItems; // all new items, including group
1837 std::vector<BOARD_ITEM*> selectedItems; // the group, or newItems if no group
1838 PCB_SELECTION preview;
1839 BOARD_COMMIT commit( m_frame );
1840 PCB_GROUP* group = nullptr;
1841 PICKED_ITEMS_LIST groupUndoList;
1842 PCB_LAYER_ID layer = F_Cu;
1843
1844 if( dlg.ShouldGroupItems() )
1845 {
1846 group = new PCB_GROUP( m_frame->GetModel() );
1847
1848 newItems.push_back( group );
1849 selectedItems.push_back( group );
1850 preview.Add( group );
1851 }
1852
1853 if( dlg.ShouldFixDiscontinuities() )
1854 {
1855 std::vector<PCB_SHAPE*> shapeList;
1856 std::vector<std::unique_ptr<PCB_SHAPE>> newShapes;
1857
1858 for( const std::unique_ptr<EDA_ITEM>& ptr : list )
1859 {
1860 if( PCB_SHAPE* shape = dynamic_cast<PCB_SHAPE*>( ptr.get() ) )
1861 shapeList.push_back( shape );
1862 }
1863
1864 ConnectBoardShapes( shapeList, newShapes, dlg.GetTolerance() );
1865
1866 for( std::unique_ptr<PCB_SHAPE>& ptr : newShapes )
1867 {
1868 ptr->SetParent( m_frame->GetBoard() );
1869 list.push_back( std::move( ptr ) );
1870 }
1871 }
1872
1873 for( std::unique_ptr<EDA_ITEM>& ptr : list )
1874 {
1875 EDA_ITEM* eda_item = ptr.release();
1876
1877 if( eda_item->IsBOARD_ITEM() )
1878 {
1879 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( eda_item );
1880
1881 newItems.push_back( item );
1882
1883 if( group )
1884 {
1885 group->AddItem( item );
1886 groupUndoList.PushItem( ITEM_PICKER( nullptr, item, UNDO_REDO::REGROUP ) );
1887 }
1888 else
1889 {
1890 selectedItems.push_back( item );
1891 }
1892
1893 layer = item->GetLayer();
1894 }
1895
1896 preview.Add( eda_item );
1897 }
1898
1899 // Clear the current selection then select the drawings so that edit tools work on them
1901
1902 EDA_ITEMS selItems( selectedItems.begin(), selectedItems.end() );
1904
1905 if( !dlg.IsPlacementInteractive() )
1906 {
1907 for( BOARD_ITEM* item : newItems )
1908 commit.Add( item );
1909
1910 if( groupUndoList.GetCount() > 0 )
1911 commit.Stage( groupUndoList );
1912
1913 commit.Push( _( "Import Graphics" ) );
1914
1915 return 0;
1916 }
1917
1918 // Turn shapes on if they are off, so that the created object will be visible after completion
1920
1921 if( !m_view->IsLayerVisible( layer ) )
1922 {
1923 m_frame->GetAppearancePanel()->SetLayerVisible( layer, true );
1925 }
1926
1927 m_view->Add( &preview );
1928
1929 m_frame->PushTool( aEvent );
1930
1931 auto setCursor =
1932 [&]()
1933 {
1934 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::MOVING );
1935 };
1936
1937 Activate();
1938 // Must be done after Activate() so that it gets set into the correct context
1939 m_controls->ShowCursor( true );
1941 // Set initial cursor
1942 setCursor();
1943
1944 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::DXF );
1946
1947 // Now move the new items to the current cursor position:
1948 VECTOR2I cursorPos = m_controls->GetCursorPosition( !aEvent.DisableGridSnapping() );
1949 VECTOR2I delta = cursorPos - static_cast<BOARD_ITEM*>( preview.GetTopLeftItem() )->GetPosition();
1950
1951 for( BOARD_ITEM* item : selectedItems )
1952 item->Move( delta );
1953
1954 m_view->Update( &preview );
1955
1956 // Main loop: keep receiving events
1957 while( TOOL_EVENT* evt = Wait() )
1958 {
1959 setCursor();
1960
1961 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
1962 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
1963 cursorPos = GetClampedCoords(
1964 grid.BestSnapAnchor( m_controls->GetMousePosition(), { layer }, GRID_GRAPHICS ),
1966 m_controls->ForceCursorPosition( true, cursorPos );
1967
1968 if( evt->IsCancelInteractive() || evt->IsActivate() )
1969 {
1971
1972 if( group )
1973 {
1974 preview.Remove( group );
1975 group->RemoveAll();
1976 }
1977
1978 for( BOARD_ITEM* item : newItems )
1979 delete item;
1980
1981 break;
1982 }
1983 else if( evt->IsMotion() )
1984 {
1985 delta = cursorPos - static_cast<BOARD_ITEM*>( preview.GetTopLeftItem() )->GetPosition();
1986
1987 for( BOARD_ITEM* item : selectedItems )
1988 item->Move( delta );
1989
1990 m_view->Update( &preview );
1991 }
1992 else if( evt->IsClick( BUT_RIGHT ) )
1993 {
1994 m_menu->ShowContextMenu( selection() );
1995 }
1996 else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
1997 {
1998 // Place the imported drawings
1999 for( BOARD_ITEM* item : newItems )
2000 commit.Add( item );
2001
2002 if( groupUndoList.GetCount() > 0 )
2003 commit.Stage( groupUndoList );
2004
2005 commit.Push( _( "Import Graphics" ) );
2006
2007 break; // This is a one-shot command, not a tool
2008 }
2009 else if( ZONE_FILLER_TOOL::IsZoneFillAction( evt ) )
2010 {
2011 wxBell();
2012 }
2013 else
2014 {
2015 evt->SetPassEvent();
2016 }
2017 }
2018
2019 preview.Clear();
2020 m_view->Remove( &preview );
2021
2022 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
2024
2025 m_frame->PopTool( aEvent );
2026
2027 return 0;
2028}
2029
2030
2032{
2033 // Make sense only in FP editor
2034 if( !m_isFootprintEditor )
2035 return 0;
2036
2037 if( !m_frame->GetModel() )
2038 return 0;
2039
2040 if( m_inDrawingTool )
2041 return 0;
2042
2044
2045 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::ANCHOR );
2047
2049
2050 m_frame->PushTool( aEvent );
2051
2052 auto setCursor =
2053 [&]()
2054 {
2055 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::BULLSEYE );
2056 };
2057
2058 Activate();
2059 // Must be done after Activate() so that it gets set into the correct context
2060 m_controls->ShowCursor( true );
2061 m_controls->SetAutoPan( true );
2062 m_controls->CaptureCursor( false );
2064 // Set initial cursor
2065 setCursor();
2066
2067 while( TOOL_EVENT* evt = Wait() )
2068 {
2069 setCursor();
2070
2071 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
2072 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
2073 VECTOR2I cursorPos = grid.BestSnapAnchor( m_controls->GetMousePosition(),
2075 m_controls->ForceCursorPosition( true, cursorPos );
2076
2077 if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
2078 {
2080 BOARD_COMMIT commit( m_frame );
2081 commit.Modify( footprint );
2082
2083 // set the new relative internal local coordinates of footprint items
2084 VECTOR2I moveVector = footprint->GetPosition() - cursorPos;
2085 footprint->MoveAnchorPosition( moveVector );
2086
2087 commit.Push( _( "Move Footprint Anchor" ) );
2088
2089 // Usually, we do not need to change twice the anchor position,
2090 // so deselect the active tool
2091 m_frame->PopTool( aEvent );
2092 break;
2093 }
2094 else if( evt->IsClick( BUT_RIGHT ) )
2095 {
2096 m_menu->ShowContextMenu( selection() );
2097 }
2098 else if( evt->IsCancelInteractive() || evt->IsActivate() )
2099 {
2100 m_frame->PopTool( aEvent );
2101 break;
2102 }
2103 else
2104 {
2105 evt->SetPassEvent();
2106 }
2107 }
2108
2109 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
2111
2112 return 0;
2113}
2114
2115
2117{
2118#define TOGGLE( a ) a = !a
2119
2121
2122 if( frame()->IsType( FRAME_PCB_EDITOR ) )
2124 else
2126
2128
2129 return 0;
2130
2131#undef TOGGLE
2132}
2133
2134
2139 PCB_SHAPE* aGraphic )
2140{
2141 if( !aMgr.IsReset() )
2142 {
2143 aGraphic->SetStart( aMgr.GetOrigin() );
2144 aGraphic->SetEnd( aMgr.GetEnd() );
2145 }
2146}
2147
2148
2149bool DRAWING_TOOL::drawShape( const TOOL_EVENT& aTool, PCB_SHAPE** aGraphic,
2150 std::optional<VECTOR2D> aStartingPoint,
2151 std::stack<PCB_SHAPE*>* aCommittedGraphics )
2152{
2153 SHAPE_T shape = ( *aGraphic )->GetShape();
2154
2155 // Only three shapes are currently supported
2156 wxASSERT( shape == SHAPE_T::SEGMENT || shape == SHAPE_T::CIRCLE || shape == SHAPE_T::RECTANGLE );
2157
2159 EDA_UNITS userUnits = m_frame->GetUserUnits();
2161 PCB_SHAPE*& graphic = *aGraphic;
2162
2163 if( m_layer != m_frame->GetActiveLayer() )
2164 {
2167 m_stroke.SetLineStyle( LINE_STYLE::DEFAULT );
2168 m_stroke.SetColor( COLOR4D::UNSPECIFIED );
2169
2178 }
2179
2180 // Turn shapes on if they are off, so that the created object will be visible after completion
2182
2183 // geometric construction manager
2185
2186 // drawing assistant overlay
2187 // TODO: workaround because EDA_SHAPE_TYPE_T is not visible from commons.
2188 KIGFX::PREVIEW::GEOM_SHAPE geomShape( static_cast<KIGFX::PREVIEW::GEOM_SHAPE>( shape ) );
2189 KIGFX::PREVIEW::TWO_POINT_ASSISTANT twoPointAsst( twoPointMgr, pcbIUScale, userUnits, geomShape );
2190
2191 // Add a VIEW_GROUP that serves as a preview for the new item
2192 m_preview.Clear();
2193 m_view->Add( &m_preview );
2194 m_view->Add( &twoPointAsst );
2195
2196 bool started = false;
2197 bool cancelled = false;
2198 bool isLocalOriginSet = ( m_frame->GetScreen()->m_LocalOrigin != VECTOR2D( 0, 0 ) );
2199 VECTOR2I cursorPos = m_controls->GetMousePosition();
2200
2201 auto setCursor =
2202 [&]()
2203 {
2204 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::PENCIL );
2205 };
2206
2207 auto cleanup =
2208 [&]()
2209 {
2210 m_preview.Clear();
2211 m_view->Update( &m_preview );
2212 delete graphic;
2213 graphic = nullptr;
2214
2215 if( !isLocalOriginSet )
2217 };
2218
2219 m_controls->ShowCursor( true );
2221 // Set initial cursor
2222 setCursor();
2223
2225
2226 if( aStartingPoint )
2227 m_toolMgr->PrimeTool( *aStartingPoint );
2228
2229 // Main loop: keep receiving events
2230 while( TOOL_EVENT* evt = Wait() )
2231 {
2232 setCursor();
2233
2234 if( started )
2235 m_frame->SetMsgPanel( graphic );
2236
2237 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
2238 bool is45Limited = Is45Limited() && !evt->Modifier( MD_CTRL );
2239
2240 // Rectangular shapes never get 45-degree snapping
2241 if( shape == SHAPE_T::RECTANGLE )
2242 is45Limited = false;
2243
2244 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
2245 cursorPos = GetClampedCoords(
2246 grid.BestSnapAnchor( m_controls->GetMousePosition(), { m_layer }, GRID_GRAPHICS ),
2248 m_controls->ForceCursorPosition( true, cursorPos );
2249
2250 if( evt->IsCancelInteractive() || ( started && evt->IsAction( &ACTIONS::undo ) ) )
2251 {
2252 cleanup();
2253
2254 if( !started )
2255 {
2256 // We've handled the cancel event. Don't cancel other tools
2257 evt->SetPassEvent( false );
2258 m_frame->PopTool( aTool );
2259 cancelled = true;
2260 }
2261
2262 break;
2263 }
2264 else if( evt->IsActivate() )
2265 {
2266 if( evt->IsPointEditor() )
2267 {
2268 // don't exit (the point editor runs in the background)
2269 }
2270 else if( evt->IsMoveTool() )
2271 {
2272 cleanup();
2273 // leave ourselves on the stack so we come back after the move
2274 cancelled = true;
2275 break;
2276 }
2277 else
2278 {
2279 cleanup();
2280 m_frame->PopTool( aTool );
2281 cancelled = true;
2282 break;
2283 }
2284 }
2285 else if( evt->IsAction( &PCB_ACTIONS::layerChanged ) )
2286 {
2287 if( m_layer != m_frame->GetActiveLayer() )
2288 {
2291 m_stroke.SetLineStyle( LINE_STYLE::DEFAULT );
2292 m_stroke.SetColor( COLOR4D::UNSPECIFIED );
2293
2302 }
2303
2304 if( graphic )
2305 {
2306 if( !m_view->IsLayerVisible( m_layer ) )
2307 {
2310 }
2311
2312 graphic->SetLayer( m_layer );
2313 graphic->SetStroke( m_stroke );
2314
2315 if( PCB_TEXTBOX* pcb_textbox = dynamic_cast<PCB_TEXTBOX*>( graphic ) )
2316 pcb_textbox->SetAttributes( m_textAttrs );
2317
2318 m_view->Update( &m_preview );
2319 frame()->SetMsgPanel( graphic );
2320 }
2321 else
2322 {
2323 evt->SetPassEvent();
2324 }
2325 }
2326 else if( evt->IsClick( BUT_RIGHT ) )
2327 {
2328 if( !graphic )
2330
2331 m_menu->ShowContextMenu( selection() );
2332 }
2333 else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
2334 {
2335 if( !graphic )
2336 break;
2337
2338 if( !started )
2339 {
2341
2342 if( aStartingPoint )
2343 {
2344 cursorPos = *aStartingPoint;
2345 aStartingPoint = std::nullopt;
2346 }
2347
2348 // Init the new item attributes
2349 if( graphic ) // always true, but Coverity can't seem to figure that out
2350 {
2351 graphic->SetShape( static_cast<SHAPE_T>( shape ) );
2352 graphic->SetFilled( false );
2353 graphic->SetStroke( m_stroke );
2354 graphic->SetLayer( m_layer );
2355 }
2356
2357 if( PCB_TEXTBOX* pcb_textbox = dynamic_cast<PCB_TEXTBOX*>( graphic ) )
2358 pcb_textbox->SetAttributes( m_textAttrs );
2359
2360 grid.SetSkipPoint( cursorPos );
2361
2362 twoPointMgr.SetOrigin( cursorPos );
2363 twoPointMgr.SetEnd( cursorPos );
2364
2365 if( !isLocalOriginSet )
2366 m_frame->GetScreen()->m_LocalOrigin = cursorPos;
2367
2368 m_preview.Add( graphic );
2369 frame()->SetMsgPanel( graphic );
2370 m_controls->SetAutoPan( true );
2371 m_controls->CaptureCursor( true );
2372
2373 if( !m_view->IsLayerVisible( m_layer ) )
2374 {
2377 }
2378
2379 updateSegmentFromGeometryMgr( twoPointMgr, graphic );
2380
2381 started = true;
2382 }
2383 else
2384 {
2385 PCB_SHAPE* snapItem = dynamic_cast<PCB_SHAPE*>( grid.GetSnapped() );
2386
2387 if( shape == SHAPE_T::SEGMENT && snapItem && graphic->GetLength() > 0 )
2388 {
2389 // User has clicked on the end of an existing segment, closing a path
2390 BOARD_COMMIT commit( m_frame );
2391
2392 commit.Add( graphic );
2393 commit.Push( _( "Draw Line" ) );
2395
2396 graphic = nullptr;
2397 }
2398 else if( twoPointMgr.IsEmpty() || evt->IsDblClick( BUT_LEFT ) )
2399 {
2400 // User has clicked twice in the same spot, meaning we're finished
2401 delete graphic;
2402 graphic = nullptr;
2403 }
2404
2405 m_preview.Clear();
2406 twoPointMgr.Reset();
2407 break;
2408 }
2409
2410 twoPointMgr.SetEnd( GetClampedCoords( cursorPos ) );
2411 }
2412 else if( evt->IsMotion() )
2413 {
2414 VECTOR2I clampedCursorPos = cursorPos;
2415
2416 if( shape == SHAPE_T::CIRCLE || shape == SHAPE_T::ARC )
2417 clampedCursorPos = getClampedRadiusEnd( twoPointMgr.GetOrigin(), cursorPos );
2418 else
2419 clampedCursorPos = getClampedDifferenceEnd( twoPointMgr.GetOrigin(), cursorPos );
2420
2421 // 45 degree lines
2422 if( started && is45Limited )
2423 {
2424 const VECTOR2I lineVector( clampedCursorPos - VECTOR2I( twoPointMgr.GetOrigin() ) );
2425
2426 // get a restricted 45/H/V line from the last fixed point to the cursor
2427 VECTOR2I newEnd = GetVectorSnapped45( lineVector, ( shape == SHAPE_T::RECTANGLE ) );
2428 m_controls->ForceCursorPosition( true, VECTOR2I( twoPointMgr.GetEnd() ) );
2429 twoPointMgr.SetEnd( twoPointMgr.GetOrigin() + newEnd );
2430 twoPointMgr.SetAngleSnap( true );
2431 }
2432 else
2433 {
2434 twoPointMgr.SetEnd( clampedCursorPos );
2435 twoPointMgr.SetAngleSnap( false );
2436 }
2437
2438 updateSegmentFromGeometryMgr( twoPointMgr, graphic );
2439 m_view->Update( &m_preview );
2440 m_view->Update( &twoPointAsst );
2441 }
2442 else if( started && ( evt->IsAction( &PCB_ACTIONS::doDelete )
2443 || evt->IsAction( &PCB_ACTIONS::deleteLastPoint ) ) )
2444 {
2445 if( aCommittedGraphics && !aCommittedGraphics->empty() )
2446 {
2447 twoPointMgr.SetOrigin( aCommittedGraphics->top()->GetStart() );
2448 twoPointMgr.SetEnd( aCommittedGraphics->top()->GetEnd() );
2449 aCommittedGraphics->pop();
2450
2451 getViewControls()->WarpMouseCursor( twoPointMgr.GetEnd(), true );
2452
2454 {
2457 delete undo;
2458 }
2459
2460 updateSegmentFromGeometryMgr( twoPointMgr, graphic );
2461 m_view->Update( &m_preview );
2462 m_view->Update( &twoPointAsst );
2463 }
2464 else
2465 {
2466 cleanup();
2467 break;
2468 }
2469 }
2470 else if( graphic && evt->IsAction( &PCB_ACTIONS::incWidth ) )
2471 {
2473 graphic->SetStroke( m_stroke );
2474 m_view->Update( &m_preview );
2475 frame()->SetMsgPanel( graphic );
2476 }
2477 else if( graphic && evt->IsAction( &PCB_ACTIONS::decWidth ) )
2478 {
2479 if( (unsigned) m_stroke.GetWidth() > WIDTH_STEP )
2480 {
2482 graphic->SetStroke( m_stroke );
2483 m_view->Update( &m_preview );
2484 frame()->SetMsgPanel( graphic );
2485 }
2486 }
2487 else if( started && evt->IsAction( &PCB_ACTIONS::properties ) )
2488 {
2489 frame()->OnEditItemRequest( graphic );
2490 m_view->Update( &m_preview );
2491 frame()->SetMsgPanel( graphic );
2492 }
2493 else if( started && ( ZONE_FILLER_TOOL::IsZoneFillAction( evt )
2494 || evt->IsAction( &ACTIONS::redo ) ) )
2495 {
2496 wxBell();
2497 }
2498 else if( evt->IsAction( &ACTIONS::resetLocalCoords ) )
2499 {
2500 isLocalOriginSet = true;
2501 evt->SetPassEvent();
2502 }
2503 else if( evt->IsAction( &ACTIONS::updateUnits ) )
2504 {
2505 if( frame()->GetUserUnits() != userUnits )
2506 {
2507 userUnits = frame()->GetUserUnits();
2508 twoPointAsst.SetUnits( userUnits );
2509 m_view->Update( &twoPointAsst );
2510 }
2511 evt->SetPassEvent();
2512 }
2513 else
2514 {
2515 evt->SetPassEvent();
2516 }
2517 }
2518
2519 if( !isLocalOriginSet ) // reset the relative coordinate if it was not set before
2521
2522 m_view->Remove( &twoPointAsst );
2523 m_view->Remove( &m_preview );
2524
2525 if( selection().Empty() )
2527
2528 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
2529 m_controls->SetAutoPan( false );
2530 m_controls->CaptureCursor( false );
2532
2533 return !cancelled;
2534}
2535
2536
2541 PCB_SHAPE& aArc )
2542{
2543 VECTOR2I vec = aMgr.GetOrigin();
2544
2545 aArc.SetCenter( vec );
2546
2547 if( aMgr.GetSubtended() < ANGLE_0 )
2548 {
2549 vec = aMgr.GetStartRadiusEnd();
2550 aArc.SetStart( vec );
2551 vec = aMgr.GetEndRadiusEnd();
2552 aArc.SetEnd( vec );
2553 }
2554 else
2555 {
2556 vec = aMgr.GetEndRadiusEnd();
2557 aArc.SetStart( vec );
2558 vec = aMgr.GetStartRadiusEnd();
2559 aArc.SetEnd( vec );
2560 }
2561}
2562
2563
2564bool DRAWING_TOOL::drawArc( const TOOL_EVENT& aTool, PCB_SHAPE** aGraphic,
2565 std::optional<VECTOR2D> aStartingPoint )
2566{
2567 wxCHECK( aGraphic, false );
2568
2569 PCB_SHAPE*& graphic = *aGraphic;
2570
2571 wxCHECK( graphic, false );
2572
2573 if( m_layer != m_frame->GetActiveLayer() )
2574 {
2577 m_stroke.SetLineStyle( LINE_STYLE::DEFAULT );
2578 m_stroke.SetColor( COLOR4D::UNSPECIFIED );
2579 }
2580
2581 // Turn shapes on if they are off, so that the created object will be visible after completion
2583
2584 // Arc geometric construction manager
2586
2587 // Arc drawing assistant overlay
2589
2590 // Add a VIEW_GROUP that serves as a preview for the new item
2591 PCB_SELECTION preview;
2592 m_view->Add( &preview );
2593 m_view->Add( &arcAsst );
2595
2596 auto setCursor =
2597 [&]()
2598 {
2599 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::PENCIL );
2600 };
2601
2602 auto cleanup =
2603 [&] ()
2604 {
2605 preview.Clear();
2606 delete *aGraphic;
2607 *aGraphic = nullptr;
2608 };
2609
2610 m_controls->ShowCursor( true );
2612 // Set initial cursor
2613 setCursor();
2614
2615 bool started = false;
2616 bool cancelled = false;
2617
2619
2620 if( aStartingPoint )
2621 m_toolMgr->PrimeTool( *aStartingPoint );
2622
2623 // Main loop: keep receiving events
2624 while( TOOL_EVENT* evt = Wait() )
2625 {
2626 if( started )
2627 m_frame->SetMsgPanel( graphic );
2628
2629 setCursor();
2630
2631 graphic->SetLayer( m_layer );
2632
2633 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
2634 bool is45Limited = Is45Limited() && !evt->Modifier( MD_CTRL );
2635 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
2636 VECTOR2I cursorPos = GetClampedCoords(
2637 grid.BestSnapAnchor( m_controls->GetMousePosition(), graphic, GRID_GRAPHICS ),
2639 m_controls->ForceCursorPosition( true, cursorPos );
2640
2641 if( evt->IsCancelInteractive() || ( started && evt->IsAction( &ACTIONS::undo ) ) )
2642 {
2643 cleanup();
2644
2645 if( !started )
2646 {
2647 // We've handled the cancel event. Don't cancel other tools
2648 evt->SetPassEvent( false );
2649 m_frame->PopTool( aTool );
2650 cancelled = true;
2651 }
2652
2653 break;
2654 }
2655 else if( evt->IsActivate() )
2656 {
2657 if( evt->IsPointEditor() )
2658 {
2659 // don't exit (the point editor runs in the background)
2660 }
2661 else if( evt->IsMoveTool() )
2662 {
2663 cleanup();
2664 // leave ourselves on the stack so we come back after the move
2665 cancelled = true;
2666 break;
2667 }
2668 else
2669 {
2670 cleanup();
2671 m_frame->PopTool( aTool );
2672 cancelled = true;
2673 break;
2674 }
2675 }
2676 else if( evt->IsClick( BUT_LEFT ) )
2677 {
2678 if( !started )
2679 {
2681
2682 m_controls->SetAutoPan( true );
2683 m_controls->CaptureCursor( true );
2684
2685 // Init the new item attributes
2686 // (non-geometric, those are handled by the manager)
2687 graphic->SetShape( SHAPE_T::ARC );
2688 graphic->SetStroke( m_stroke );
2689
2690 if( !m_view->IsLayerVisible( m_layer ) )
2691 {
2694 }
2695
2696 preview.Add( graphic );
2697 frame()->SetMsgPanel( graphic );
2698 started = true;
2699 }
2700
2701 arcManager.AddPoint( cursorPos, true );
2702 }
2703 else if( evt->IsAction( &PCB_ACTIONS::deleteLastPoint ) )
2704 {
2705 arcManager.RemoveLastPoint();
2706 }
2707 else if( evt->IsMotion() )
2708 {
2709 // set angle snap
2710 arcManager.SetAngleSnap( is45Limited );
2711
2712 // update, but don't step the manager state
2713 arcManager.AddPoint( cursorPos, false );
2714 }
2715 else if( evt->IsAction( &PCB_ACTIONS::layerChanged ) )
2716 {
2717 if( m_layer != m_frame->GetActiveLayer() )
2718 {
2721 m_stroke.SetLineStyle( LINE_STYLE::DEFAULT );
2722 m_stroke.SetColor( COLOR4D::UNSPECIFIED );
2723 }
2724
2725 if( graphic )
2726 {
2727 if( !m_view->IsLayerVisible( m_layer ) )
2728 {
2731 }
2732
2733 graphic->SetLayer( m_layer );
2734 graphic->SetStroke( m_stroke );
2735 m_view->Update( &preview );
2736 frame()->SetMsgPanel( graphic );
2737 }
2738 else
2739 {
2740 evt->SetPassEvent();
2741 }
2742 }
2743 else if( evt->IsAction( &PCB_ACTIONS::properties ) )
2744 {
2746 {
2747 graphic->SetArcAngleAndEnd( ANGLE_90 );
2748 frame()->OnEditItemRequest( graphic );
2749 m_view->Update( &preview );
2750 frame()->SetMsgPanel( graphic );
2751 break;
2752 }
2753 // Don't show the edit panel if we can't represent the arc with it
2754 else if( ( arcManager.GetStep() == KIGFX::PREVIEW::ARC_GEOM_MANAGER::SET_ANGLE )
2755 && ( arcManager.GetStartRadiusEnd() != arcManager.GetEndRadiusEnd() ) )
2756 {
2757 frame()->OnEditItemRequest( graphic );
2758 m_view->Update( &preview );
2759 frame()->SetMsgPanel( graphic );
2760 break;
2761 }
2762 else
2763 {
2764 evt->SetPassEvent();
2765 }
2766 }
2767 else if( evt->IsClick( BUT_RIGHT ) )
2768 {
2769 if( !graphic )
2771
2772 m_menu->ShowContextMenu( selection() );
2773 }
2774 else if( evt->IsAction( &PCB_ACTIONS::incWidth ) )
2775 {
2777
2778 if( graphic )
2779 {
2780 graphic->SetStroke( m_stroke );
2781 m_view->Update( &preview );
2782 frame()->SetMsgPanel( graphic );
2783 }
2784 }
2785 else if( evt->IsAction( &PCB_ACTIONS::decWidth ) )
2786 {
2787 if( (unsigned) m_stroke.GetWidth() > WIDTH_STEP )
2788 {
2790
2791 if( graphic )
2792 {
2793 graphic->SetStroke( m_stroke );
2794 m_view->Update( &preview );
2795 frame()->SetMsgPanel( graphic );
2796 }
2797 }
2798 }
2799 else if( evt->IsAction( &PCB_ACTIONS::arcPosture ) )
2800 {
2801 arcManager.ToggleClockwise();
2802 }
2803 else if( evt->IsAction( &ACTIONS::updateUnits ) )
2804 {
2805 arcAsst.SetUnits( frame()->GetUserUnits() );
2806 m_view->Update( &arcAsst );
2807 evt->SetPassEvent();
2808 }
2809 else if( started && ( ZONE_FILLER_TOOL::IsZoneFillAction( evt )
2810 || evt->IsAction( &ACTIONS::redo ) ) )
2811 {
2812 wxBell();
2813 }
2814 else
2815 {
2816 evt->SetPassEvent();
2817 }
2818
2819 if( arcManager.IsComplete() )
2820 {
2821 break;
2822 }
2823 else if( arcManager.HasGeometryChanged() )
2824 {
2825 updateArcFromConstructionMgr( arcManager, *graphic );
2826 m_view->Update( &preview );
2827 m_view->Update( &arcAsst );
2828
2829 if( started )
2830 frame()->SetMsgPanel( graphic );
2831 else
2832 frame()->SetMsgPanel( board() );
2833 }
2834 }
2835
2836 preview.Remove( graphic );
2837 m_view->Remove( &arcAsst );
2838 m_view->Remove( &preview );
2839
2840 if( selection().Empty() )
2842
2843 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
2844 m_controls->SetAutoPan( false );
2845 m_controls->CaptureCursor( false );
2847
2848 return !cancelled;
2849}
2850
2851
2856 PCB_SHAPE& aBezier )
2857{
2858 VECTOR2I vec = aMgr.GetStart();
2859
2860 aBezier.SetStart( vec );
2861 aBezier.SetBezierC1( aMgr.GetControlC1() );
2862 aBezier.SetEnd( aMgr.GetEnd() );
2863 aBezier.SetBezierC2( aMgr.GetControlC2() );
2864
2865 // Need this for the length preview to work
2867}
2868
2869
2870std::unique_ptr<PCB_SHAPE> DRAWING_TOOL::drawOneBezier( const TOOL_EVENT& aTool,
2871 const OPT_VECTOR2I& aStartingPoint,
2872 const OPT_VECTOR2I& aStartingControl1Point,
2873 DRAW_ONE_RESULT& aResult )
2874{
2875 std::unique_ptr<PCB_SHAPE> bezier = std::make_unique<PCB_SHAPE>( m_frame->GetModel() );
2876 bezier->SetShape( SHAPE_T::BEZIER );
2877 bezier->SetFlags( IS_NEW );
2878
2879 if( m_layer != m_frame->GetActiveLayer() )
2880 {
2883 m_stroke.SetLineStyle( LINE_STYLE::DEFAULT );
2884 m_stroke.SetColor( COLOR4D::UNSPECIFIED );
2885 }
2886
2887 // Turn shapes on if they are off, so that the created object will be visible after completion
2889
2890 // Arc geometric construction manager
2892
2893 // Arc drawing assistant overlay
2894 KIGFX::PREVIEW::BEZIER_ASSISTANT bezierAsst( bezierManager, pcbIUScale,
2895 m_frame->GetUserUnits() );
2896
2897 // Add a VIEW_GROUP that serves as a preview for the new item
2898 PCB_SELECTION preview;
2899 m_view->Add( &preview );
2900 m_view->Add( &bezierAsst );
2902
2903 const auto setCursor = [&]()
2904 {
2905 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::PENCIL );
2906 };
2907
2908 const auto resetProgress = [&]()
2909 {
2910 preview.Clear();
2911 bezier.reset();
2912 };
2913
2914 m_controls->ShowCursor( true );
2916 // Set initial cursor
2917 setCursor();
2918
2919 const auto started = [&]()
2920 {
2922 };
2923
2924 aResult = DRAW_ONE_RESULT::ACCEPTED;
2925 bool priming = false;
2926
2928
2929 // Load in one or two points if they were passed in
2930 if( aStartingPoint )
2931 {
2932 priming = true;
2933
2934 if( aStartingControl1Point )
2935 {
2936 bezierManager.AddPoint( *aStartingPoint, true );
2937 bezierManager.AddPoint( *aStartingControl1Point, true );
2938 m_toolMgr->PrimeTool( *aStartingControl1Point );
2939 }
2940 else
2941 {
2942 bezierManager.AddPoint( *aStartingPoint, true );
2943 m_toolMgr->PrimeTool( *aStartingPoint );
2944 }
2945 }
2946
2947 // Main loop: keep receiving events
2948 while( TOOL_EVENT* evt = Wait() )
2949 {
2950 if( started() )
2951 m_frame->SetMsgPanel( bezier.get() );
2952
2953 setCursor();
2954
2955 // Init the new item attributes
2956 // (non-geometric, those are handled by the manager)
2957 bezier->SetShape( SHAPE_T::BEZIER );
2958 bezier->SetStroke( m_stroke );
2959 bezier->SetLayer( m_layer );
2960
2961 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
2962 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
2963 VECTOR2I cursorPos = GetClampedCoords(
2964 grid.BestSnapAnchor( m_controls->GetMousePosition(), bezier.get(), GRID_GRAPHICS ),
2966 m_controls->ForceCursorPosition( true, cursorPos );
2967
2968 if( evt->IsCancelInteractive() || ( started() && evt->IsAction( &ACTIONS::undo ) ) )
2969 {
2970 resetProgress();
2971
2972 if( !started() )
2973 {
2974 // We've handled the cancel event. Don't cancel other tools
2975 evt->SetPassEvent( false );
2976 m_frame->PopTool( aTool );
2978 }
2979 else
2980 {
2981 // We're not cancelling, but we're also not returning a finished bezier
2982 // So we'll be called again.
2983 aResult = DRAW_ONE_RESULT::RESET;
2984 }
2985
2986 break;
2987 }
2988 else if( evt->IsActivate() )
2989 {
2990 if( evt->IsPointEditor() )
2991 {
2992 // don't exit (the point editor runs in the background)
2993 }
2994 else if( evt->IsMoveTool() )
2995 {
2996 resetProgress();
2997 // leave ourselves on the stack so we come back after the move
2999 break;
3000 }
3001 else
3002 {
3003 resetProgress();
3004 m_frame->PopTool( aTool );
3006 break;
3007 }
3008 }
3009 else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
3010 {
3011 if( !started() )
3012 {
3014
3015 m_controls->SetAutoPan( true );
3016 m_controls->CaptureCursor( true );
3017
3018 if( !m_view->IsLayerVisible( m_layer ) )
3019 {
3022 }
3023
3024 frame()->SetMsgPanel( bezier.get() );
3025 }
3026
3027 if( !priming )
3028 bezierManager.AddPoint( cursorPos, true );
3029 else
3030 priming = false;
3031
3032 const bool doubleClick = evt->IsDblClick( BUT_LEFT );
3033
3034 if( doubleClick )
3035 {
3036 // Use the current point for all remaining points
3037 while( bezierManager.GetStep() < KIGFX::PREVIEW::BEZIER_GEOM_MANAGER::SET_END )
3038 {
3039 bezierManager.AddPoint( cursorPos, true );
3040 }
3041 }
3042
3044 {
3045 preview.Add( bezier.get() );
3046 }
3047
3048 // Return to the caller for a reset
3049 if( doubleClick )
3050 {
3051 // Don't chain to this one
3053 break;
3054 }
3055 }
3056 else if( evt->IsAction( &PCB_ACTIONS::deleteLastPoint ) )
3057 {
3058 bezierManager.RemoveLastPoint();
3059
3061 {
3062 preview.Remove( bezier.get() );
3063 }
3064 }
3065 else if( evt->IsMotion() )
3066 {
3067 // set angle snap
3068 // bezierManager.SetAngleSnap( Is45Limited() );
3069
3070 // update, but don't step the manager state
3071 bezierManager.AddPoint( cursorPos, false );
3072 }
3073 else if( evt->IsAction( &PCB_ACTIONS::layerChanged ) )
3074 {
3075 if( m_layer != m_frame->GetActiveLayer() )
3076 {
3079 m_stroke.SetLineStyle( LINE_STYLE::DEFAULT );
3080 m_stroke.SetColor( COLOR4D::UNSPECIFIED );
3081 }
3082
3083 if( bezier )
3084 {
3085 if( !m_view->IsLayerVisible( m_layer ) )
3086 {
3089 }
3090
3091 bezier->SetLayer( m_layer );
3092 bezier->SetStroke( m_stroke );
3093 m_view->Update( &preview );
3094 frame()->SetMsgPanel( bezier.get() );
3095 }
3096 else
3097 {
3098 evt->SetPassEvent();
3099 }
3100 }
3101 else if( evt->IsAction( &PCB_ACTIONS::properties ) )
3102 {
3103 // Don't show the edit panel if we can't represent the arc with it
3104 if( ( bezierManager.GetStep() >= KIGFX::PREVIEW::BEZIER_GEOM_MANAGER::SET_END ) )
3105 {
3106 frame()->OnEditItemRequest( bezier.get() );
3107 m_view->Update( &preview );
3108 frame()->SetMsgPanel( bezier.get() );
3109 break;
3110 }
3111 else
3112 {
3113 evt->SetPassEvent();
3114 }
3115 }
3116 else if( evt->IsClick( BUT_RIGHT ) )
3117 {
3118 if( !bezier )
3120
3121 m_menu->ShowContextMenu( selection() );
3122 }
3123 else if( evt->IsAction( &PCB_ACTIONS::incWidth ) )
3124 {
3126
3127 if( bezier )
3128 {
3129 bezier->SetStroke( m_stroke );
3130 m_view->Update( &preview );
3131 frame()->SetMsgPanel( bezier.get() );
3132 }
3133 }
3134 else if( evt->IsAction( &PCB_ACTIONS::decWidth ) )
3135 {
3136 if( (unsigned) m_stroke.GetWidth() > WIDTH_STEP )
3137 {
3139
3140 if( bezier )
3141 {
3142 bezier->SetStroke( m_stroke );
3143 m_view->Update( &preview );
3144 frame()->SetMsgPanel( bezier.get() );
3145 }
3146 }
3147 }
3148 else if( evt->IsAction( &ACTIONS::updateUnits ) )
3149 {
3150 bezierAsst.SetUnits( frame()->GetUserUnits() );
3151 m_view->Update( &bezierAsst );
3152 evt->SetPassEvent();
3153 }
3154 else if( started()
3156 || evt->IsAction( &ACTIONS::redo ) ) )
3157 {
3158 wxBell();
3159 }
3160 else
3161 {
3162 evt->SetPassEvent();
3163 }
3164
3165 if( bezierManager.IsComplete() )
3166 {
3167 break;
3168 }
3169 else if( bezierManager.HasGeometryChanged() )
3170 {
3171 updateBezierFromConstructionMgr( bezierManager, *bezier );
3172 m_view->Update( &preview );
3173 m_view->Update( &bezierAsst );
3174
3175 // Once we are receiving end points, we can show the bezier in the preview
3177 frame()->SetMsgPanel( bezier.get() );
3178 else
3179 frame()->SetMsgPanel( board() );
3180 }
3181 }
3182
3183 preview.Remove( bezier.get() );
3184 m_view->Remove( &bezierAsst );
3185 m_view->Remove( &preview );
3186
3187 if( selection().Empty() )
3189
3190 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
3191 m_controls->SetAutoPan( false );
3192 m_controls->CaptureCursor( false );
3194
3195 return bezier;
3196};
3197
3199{
3200 bool clearSelection = false;
3201 *aZone = nullptr;
3202
3203 // not an action that needs a source zone
3204 if( aMode == ZONE_MODE::ADD || aMode == ZONE_MODE::GRAPHIC_POLYGON )
3205 return true;
3206
3208 const PCB_SELECTION& selection = selTool->GetSelection();
3209
3210 if( selection.Empty() )
3211 {
3212 clearSelection = true;
3214 }
3215
3216 // we want a single zone
3217 if( selection.Size() == 1 && selection[0]->Type() == PCB_ZONE_T )
3218 *aZone = static_cast<ZONE*>( selection[0] );
3219
3220 // expected a zone, but didn't get one
3221 if( !*aZone )
3222 {
3223 if( clearSelection )
3225
3226 return false;
3227 }
3228
3229 return true;
3230}
3231
3233{
3235 return 0;
3236
3237 ZONE_MODE zoneMode = aEvent.Parameter<ZONE_MODE>();
3238 MODE drawMode = MODE::ZONE;
3239
3240 if( aEvent.IsAction( &PCB_ACTIONS::drawRuleArea ) )
3241 drawMode = MODE::KEEPOUT;
3242
3243 if( aEvent.IsAction( &PCB_ACTIONS::drawPolygon ) )
3244 drawMode = MODE::GRAPHIC_POLYGON;
3245
3246 SCOPED_DRAW_MODE scopedDrawMode( m_mode, drawMode );
3247
3248 // get a source zone, if we need one. We need it for:
3249 // ZONE_MODE::CUTOUT (adding a hole to the source zone)
3250 // ZONE_MODE::SIMILAR (creating a new zone using settings of source zone
3251 ZONE* sourceZone = nullptr;
3252
3253 if( !getSourceZoneForAction( zoneMode, &sourceZone ) )
3254 return 0;
3255
3256 // Turn zones on if they are off, so that the created object will be visible after completion
3258
3260
3261 params.m_keepout = drawMode == MODE::KEEPOUT;
3262 params.m_mode = zoneMode;
3263 params.m_sourceZone = sourceZone;
3264 params.m_layer = m_frame->GetActiveLayer();
3265
3266 if( zoneMode == ZONE_MODE::SIMILAR && !sourceZone->IsOnLayer( params.m_layer ) )
3267 params.m_layer = sourceZone->GetFirstLayer();
3268
3269 ZONE_CREATE_HELPER zoneTool( *this, params );
3270 // the geometry manager which handles the zone geometry, and hands the calculated points
3271 // over to the zone creator tool
3272 POLYGON_GEOM_MANAGER polyGeomMgr( zoneTool );
3273 bool started = false;
3275
3276 m_frame->PushTool( aEvent );
3277
3278 auto setCursor =
3279 [&]()
3280 {
3281 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::PENCIL );
3282 };
3283
3284 auto cleanup =
3285 [&] ()
3286 {
3287 polyGeomMgr.Reset();
3288 started = false;
3289 grid.ClearSkipPoint();
3290 m_controls->SetAutoPan( false );
3291 m_controls->CaptureCursor( false );
3292 };
3293
3294 Activate();
3295 // Must be done after Activate() so that it gets set into the correct context
3296 m_controls->ShowCursor( true );
3298 // Set initial cursor
3299 setCursor();
3300
3301 if( aEvent.HasPosition() )
3302 m_toolMgr->PrimeTool( aEvent.Position() );
3303
3304 // Main loop: keep receiving events
3305 while( TOOL_EVENT* evt = Wait() )
3306 {
3307 setCursor();
3308
3309 LSET layers( { m_frame->GetActiveLayer() } );
3310 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
3311 bool is45Limited = Is45Limited() && !evt->Modifier( MD_CTRL );
3312 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
3313
3314 VECTOR2I cursorPos = evt->HasPosition() ? evt->Position() : m_controls->GetMousePosition();
3315 cursorPos = GetClampedCoords( grid.BestSnapAnchor( cursorPos, layers, GRID_GRAPHICS ),
3317
3318 m_controls->ForceCursorPosition( true, cursorPos );
3319
3322
3323 if( evt->IsCancelInteractive() )
3324 {
3325 if( started )
3326 {
3327 cleanup();
3328 }
3329 else
3330 {
3331 m_frame->PopTool( aEvent );
3332
3333 // We've handled the cancel event. Don't cancel other tools
3334 evt->SetPassEvent( false );
3335 break;
3336 }
3337 }
3338 else if( evt->IsActivate() )
3339 {
3340 if( started )
3341 cleanup();
3342
3343 if( evt->IsPointEditor() )
3344 {
3345 // don't exit (the point editor runs in the background)
3346 }
3347 else if( evt->IsMoveTool() )
3348 {
3349 // leave ourselves on the stack so we come back after the move
3350 break;
3351 }
3352 else
3353 {
3354 m_frame->PopTool( aEvent );
3355 break;
3356 }
3357 }
3358 else if( evt->IsAction( &PCB_ACTIONS::layerChanged ) )
3359 {
3360 if( zoneMode != ZONE_MODE::SIMILAR )
3361 params.m_layer = frame()->GetActiveLayer();
3362
3363 if( !m_view->IsLayerVisible( params.m_layer ) )
3364 {
3367 }
3368 }
3369 else if( evt->IsClick( BUT_RIGHT ) )
3370 {
3371 if( !started )
3373
3374 m_menu->ShowContextMenu( selection() );
3375 }
3376 // events that lock in nodes
3377 else if( evt->IsClick( BUT_LEFT )
3378 || evt->IsDblClick( BUT_LEFT )
3379 || evt->IsAction( &PCB_ACTIONS::closeOutline ) )
3380 {
3381 // Check if it is double click / closing line (so we have to finish the zone)
3382 const bool endPolygon = evt->IsDblClick( BUT_LEFT )
3383 || evt->IsAction( &PCB_ACTIONS::closeOutline )
3384 || polyGeomMgr.NewPointClosesOutline( cursorPos );
3385
3386 if( endPolygon )
3387 {
3388 polyGeomMgr.SetFinished();
3389 polyGeomMgr.Reset();
3390
3391 started = false;
3392 m_controls->SetAutoPan( false );
3393 m_controls->CaptureCursor( false );
3394 }
3395 // adding a corner
3396 else if( polyGeomMgr.AddPoint( cursorPos ) )
3397 {
3398 if( !started )
3399 {
3400 started = true;
3401
3402 m_controls->SetAutoPan( true );
3403 m_controls->CaptureCursor( true );
3404
3405 if( !m_view->IsLayerVisible( params.m_layer ) )
3406 {
3409 }
3410 }
3411 }
3412 }
3413 else if( started && ( evt->IsAction( &PCB_ACTIONS::deleteLastPoint )
3414 || evt->IsAction( &ACTIONS::doDelete )
3415 || evt->IsAction( &ACTIONS::undo ) ) )
3416 {
3417 if( std::optional<VECTOR2I> last = polyGeomMgr.DeleteLastCorner() )
3418 {
3419 cursorPos = last.value();
3420 getViewControls()->WarpMouseCursor( cursorPos, true );
3421 m_controls->ForceCursorPosition( true, cursorPos );
3422 polyGeomMgr.SetCursorPosition( cursorPos );
3423 }
3424 else
3425 {
3426 cleanup();
3427 }
3428 }
3429 else if( started && ( evt->IsMotion()
3430 || evt->IsDrag( BUT_LEFT ) ) )
3431 {
3432 polyGeomMgr.SetCursorPosition( cursorPos );
3433 }
3434 else if( started && ( ZONE_FILLER_TOOL::IsZoneFillAction( evt )
3435 || evt->IsAction( &ACTIONS::redo ) ) )
3436 {
3437 wxBell();
3438 }
3439 else if( started && evt->IsAction( &PCB_ACTIONS::properties ) )
3440 {
3441 frame()->OnEditItemRequest( zoneTool.GetZone() );
3442 zoneTool.OnGeometryChange( polyGeomMgr );
3443 frame()->SetMsgPanel( zoneTool.GetZone() );
3444 }
3445 /*else if( evt->IsAction( &ACTIONS::updateUnits ) )
3446 {
3447 // If we ever have an assistant here that reports dimensions, we'll want to
3448 // update its units here....
3449 // zoneAsst.SetUnits( frame()->GetUserUnits() );
3450 // m_view->Update( &zoneAsst );
3451 evt->SetPassEvent();
3452 }*/
3453 else
3454 {
3455 evt->SetPassEvent();
3456 }
3457
3458 } // end while
3459
3460 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW );
3462 controls()->SetAutoPan( false );
3463 m_controls->CaptureCursor( false );
3464 return 0;
3465}
3466
3467
3469{
3471 return 0;
3472
3473 struct VIA_PLACER : public INTERACTIVE_PLACER_BASE
3474 {
3475 PCB_BASE_EDIT_FRAME* m_frame;
3476 PCB_GRID_HELPER m_gridHelper;
3477 std::shared_ptr<DRC_ENGINE> m_drcEngine;
3478 int m_drcEpsilon;
3479 int m_worstClearance;
3480 bool m_allowDRCViolations;
3481
3482 VIA_PLACER( PCB_BASE_EDIT_FRAME* aFrame ) :
3483 m_frame( aFrame ),
3484 m_gridHelper( aFrame->GetToolManager(), aFrame->GetMagneticItemsSettings() ),
3485 m_drcEngine( aFrame->GetBoard()->GetDesignSettings().m_DRCEngine ),
3486 m_drcEpsilon( aFrame->GetBoard()->GetDesignSettings().GetDRCEpsilon() ),
3487 m_worstClearance( 0 )
3488 {
3489 ROUTER_TOOL* router = m_frame->GetToolManager()->GetTool<ROUTER_TOOL>();
3490
3491 if( router )
3492 m_allowDRCViolations = router->Router()->Settings().AllowDRCViolations();
3493
3494 try
3495 {
3496 if( aFrame )
3497 m_drcEngine->InitEngine( aFrame->GetDesignRulesPath() );
3498
3499 DRC_CONSTRAINT constraint;
3500
3501 if( m_drcEngine->QueryWorstConstraint( CLEARANCE_CONSTRAINT, constraint ) )
3502 m_worstClearance = constraint.GetValue().Min();
3503
3504 if( m_drcEngine->QueryWorstConstraint( HOLE_CLEARANCE_CONSTRAINT, constraint ) )
3505 m_worstClearance = std::max( m_worstClearance, constraint.GetValue().Min() );
3506
3507 for( FOOTPRINT* footprint : aFrame->GetBoard()->Footprints() )
3508 {
3509 for( PAD* pad : footprint->Pads() )
3510 {
3511 std::optional<int> padOverride = pad->GetClearanceOverrides( nullptr );
3512
3513 if( padOverride.has_value() )
3514 m_worstClearance = std::max( m_worstClearance, padOverride.value() );
3515 }
3516 }
3517 }
3518 catch( PARSE_ERROR& )
3519 {
3520 }
3521 }
3522
3523 virtual ~VIA_PLACER()
3524 {
3525 }
3526
3527 PCB_TRACK* findTrack( PCB_VIA* aVia )
3528 {
3529 const LSET lset = aVia->GetLayerSet();
3530 VECTOR2I position = aVia->GetPosition();
3531 BOX2I bbox = aVia->GetBoundingBox();
3532
3533 std::vector<KIGFX::VIEW::LAYER_ITEM_PAIR> items;
3534 KIGFX::PCB_VIEW* view = m_frame->GetCanvas()->GetView();
3535 std::vector<PCB_TRACK*> possible_tracks;
3536
3537 wxCHECK( view, nullptr );
3538
3539 view->Query( bbox, items );
3540
3541 for( const KIGFX::VIEW::LAYER_ITEM_PAIR& it : items )
3542 {
3543 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( it.first );
3544
3545 if( !( item->GetLayerSet() & lset ).any() )
3546 continue;
3547
3548 if( item->Type() == PCB_TRACE_T )
3549 {
3550 PCB_TRACK* track = static_cast<PCB_TRACK*>( item );
3551
3552 if( TestSegmentHit( position, track->GetStart(), track->GetEnd(),
3553 ( track->GetWidth()
3554 + aVia->GetWidth( track->GetLayer() ) ) / 2 ) )
3555 {
3556 possible_tracks.push_back( track );
3557 }
3558 }
3559 else if( item->Type() == PCB_ARC_T )
3560 {
3561 PCB_ARC* arc = static_cast<PCB_ARC*>( item );
3562
3563 if( arc->HitTest( position, aVia->GetWidth( arc->GetLayer() ) / 2 ) )
3564 possible_tracks.push_back( arc );
3565 }
3566 }
3567
3568 PCB_TRACK* return_track = nullptr;
3569 int min_d = std::numeric_limits<int>::max();
3570
3571 for( PCB_TRACK* track : possible_tracks )
3572 {
3573 SEG test( track->GetStart(), track->GetEnd() );
3574 int dist = ( test.NearestPoint( position ) - position ).EuclideanNorm();
3575
3576 if( dist < min_d )
3577 {
3578 min_d = dist;
3579 return_track = track;
3580 }
3581 }
3582
3583 return return_track;
3584 }
3585
3586 bool hasDRCViolation( PCB_VIA* aVia, BOARD_ITEM* aOther )
3587 {
3588 DRC_CONSTRAINT constraint;
3589 int clearance;
3590 BOARD_CONNECTED_ITEM* connectedItem = dynamic_cast<BOARD_CONNECTED_ITEM*>( aOther );
3591 ZONE* zone = dynamic_cast<ZONE*>( aOther );
3592
3593 if( zone && zone->GetIsRuleArea() )
3594 {
3595 if( zone->GetDoNotAllowVias() )
3596 {
3597 bool hit = false;
3598
3600 [&]( PCB_LAYER_ID aLayer )
3601 {
3602 if( hit )
3603 return;
3604
3605 if( zone->Outline()->Collide( aVia->GetPosition(),
3606 aVia->GetWidth( aLayer ) / 2 ) )
3607 {
3608 hit = true;
3609 }
3610 } );
3611
3612 return hit;
3613 }
3614
3615 return false;
3616 }
3617
3618 if( connectedItem )
3619 {
3620 int connectedItemNet = connectedItem->GetNetCode();
3621
3622 if( connectedItemNet == 0 || connectedItemNet == aVia->GetNetCode() )
3623 return false;
3624 }
3625
3626 for( PCB_LAYER_ID layer : aOther->GetLayerSet().Seq() )
3627 {
3628 // Reference images are "on" a copper layer but are not actually part of it
3629 if( !IsCopperLayer( layer ) || aOther->Type() == PCB_REFERENCE_IMAGE_T )
3630 continue;
3631
3632 constraint = m_drcEngine->EvalRules( CLEARANCE_CONSTRAINT, aVia, aOther, layer );
3633 clearance = constraint.GetValue().Min();
3634
3635 if( clearance >= 0 )
3636 {
3637 std::shared_ptr<SHAPE> viaShape = aVia->GetEffectiveShape( layer );
3638 std::shared_ptr<SHAPE> otherShape = aOther->GetEffectiveShape( layer );
3639
3640 if( viaShape->Collide( otherShape.get(), clearance - m_drcEpsilon ) )
3641 return true;
3642 }
3643 }
3644
3645 if( aOther->HasHole() )
3646 {
3647 constraint = m_drcEngine->EvalRules( HOLE_CLEARANCE_CONSTRAINT, aVia, aOther,
3649 clearance = constraint.GetValue().Min();
3650
3651 if( clearance >= 0 )
3652 {
3653 std::shared_ptr<SHAPE> viaShape = aVia->GetEffectiveShape( UNDEFINED_LAYER );
3654
3655 if( viaShape->Collide( aOther->GetEffectiveHoleShape().get(),
3656 clearance - m_drcEpsilon ) )
3657 {
3658 return true;
3659 }
3660 }
3661 }
3662
3663 return false;
3664 }
3665
3666 bool checkDRCViolation( PCB_VIA* aVia )
3667 {
3668 std::vector<KIGFX::VIEW::LAYER_ITEM_PAIR> items;
3669 std::set<BOARD_ITEM*> checkedItems;
3670 BOX2I bbox = aVia->GetBoundingBox();
3671
3672 bbox.Inflate( m_worstClearance );
3673 m_frame->GetCanvas()->GetView()->Query( bbox, items );
3674
3675 for( std::pair<KIGFX::VIEW_ITEM*, int> it : items )
3676 {
3677 if( !it.first->IsBOARD_ITEM() )
3678 continue;
3679
3680 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( it.first );
3681
3682 if( item->Type() == PCB_ZONE_T && !static_cast<ZONE*>( item )->GetIsRuleArea() )
3683 {
3684 continue; // stitching vias bind to zones, so ignore them
3685 }
3686 else if( item->Type() == PCB_FOOTPRINT_T || item->Type() == PCB_GROUP_T )
3687 {
3688 continue; // check against children, but not against footprint itself
3689 }
3690 else if( ( item->Type() == PCB_FIELD_T || item->Type() == PCB_TEXT_T )
3691 && !static_cast<PCB_TEXT*>( item )->IsVisible() )
3692 {
3693 continue; // ignore hidden items
3694 }
3695 else if( checkedItems.count( item ) )
3696 {
3697 continue; // already checked
3698 }
3699
3700 if( hasDRCViolation( aVia, item ) )
3701 return true;
3702
3703 checkedItems.insert( item );
3704 }
3705
3706 DRC_CONSTRAINT constraint = m_drcEngine->EvalRules( DISALLOW_CONSTRAINT, aVia, nullptr,
3708
3709 if( constraint.m_DisallowFlags && constraint.GetSeverity() != RPT_SEVERITY_IGNORE )
3710 return true;
3711
3712 return false;
3713 }
3714
3715 PAD* findPad( PCB_VIA* aVia )
3716 {
3717 const VECTOR2I position = aVia->GetPosition();
3718 const LSET lset = aVia->GetLayerSet();
3719
3720 for( FOOTPRINT* fp : m_board->Footprints() )
3721 {
3722 for( PAD* pad : fp->Pads() )
3723 {
3724 if( pad->HitTest( position ) && ( pad->GetLayerSet() & lset ).any() )
3725 {
3726 return pad;
3727 }
3728 }
3729 }
3730
3731 return nullptr;
3732 }
3733
3734 PCB_SHAPE* findGraphic( const PCB_VIA* aVia ) const
3735 {
3736 const LSET lset = aVia->GetLayerSet() & LSET::AllCuMask();
3737 VECTOR2I position = aVia->GetPosition();
3738 BOX2I bbox = aVia->GetBoundingBox();
3739
3740 std::vector<KIGFX::VIEW::LAYER_ITEM_PAIR> items;
3741 KIGFX::PCB_VIEW* view = m_frame->GetCanvas()->GetView();
3742 PCB_LAYER_ID activeLayer = m_frame->GetActiveLayer();
3743 std::vector<PCB_SHAPE*> possible_shapes;
3744
3745 view->Query( bbox, items );
3746
3747 for( const KIGFX::VIEW::LAYER_ITEM_PAIR& it : items )
3748 {
3749 BOARD_ITEM* item = static_cast<BOARD_ITEM*>( it.first );
3750
3751 if( !( item->GetLayerSet() & lset ).any() )
3752 continue;
3753
3754 if( item->Type() == PCB_SHAPE_T )
3755 {
3756 PCB_SHAPE* shape = static_cast<PCB_SHAPE*>( item );
3757
3758 if( shape->HitTest( position, aVia->GetWidth( activeLayer ) / 2 ) )
3759 possible_shapes.push_back( shape );
3760 }
3761 }
3762
3763 PCB_SHAPE* return_shape = nullptr;
3764 int min_d = std::numeric_limits<int>::max();
3765
3766 for( PCB_SHAPE* shape : possible_shapes )
3767 {
3768 int dist = ( shape->GetPosition() - position ).EuclideanNorm();
3769
3770 if( dist < min_d )
3771 {
3772 min_d = dist;
3773 return_shape = shape;
3774 }
3775 }
3776
3777 return return_shape;
3778 }
3779
3780 std::optional<int> selectPossibleNetsByPopupMenu( std::set<int>& aNetcodeList )
3781 {
3782 ACTION_MENU menu( true );
3783 const NETINFO_LIST& netInfo = m_board->GetNetInfo();
3784 std::map<int, int> menuIDNetCodeMap;
3785 int menuID = 1;
3786
3787 for( int netcode : aNetcodeList )
3788 {
3789 wxString menuText;
3790 if( menuID < 10 )
3791 {
3792#ifdef __WXMAC__
3793 menuText = wxString::Format( "%s\t",
3794 netInfo.GetNetItem( netcode )->GetNetname() );
3795#else
3796 menuText = wxString::Format( "&%d %s\t",
3797 menuID,
3798 netInfo.GetNetItem( netcode )->GetNetname() );
3799#endif
3800 }
3801 else
3802 {
3803 menuText = netInfo.GetNetItem( netcode )->GetNetname();
3804 }
3805
3806 menu.Add( menuText, menuID, BITMAPS::INVALID_BITMAP );
3807 menuIDNetCodeMap[ menuID ] = netcode;
3808 menuID++;
3809 }
3810
3811 menu.SetTitle( _( "Select Net:" ) );
3812 menu.DisplayTitle( true );
3813
3814 DRAWING_TOOL* drawingTool = m_frame->GetToolManager()->GetTool<DRAWING_TOOL>();
3815 drawingTool->SetContextMenu( &menu, CMENU_NOW );
3816
3817 int selectedNetCode = -1;
3818 bool cancelled = false;
3819
3820 while( TOOL_EVENT* evt = drawingTool->Wait() )
3821 {
3822 if( evt->Action() == TA_CHOICE_MENU_UPDATE )
3823 {
3824 evt->SetPassEvent();
3825 }
3826 else if( evt->Action() == TA_CHOICE_MENU_CHOICE )
3827 {
3828 std::optional<int> id = evt->GetCommandId();
3829
3830 // User has selected an item, so this one will be returned
3831 if( id && ( *id > 0 ) && ( *id < menuID ) )
3832 {
3833 selectedNetCode = menuIDNetCodeMap.at( *id );
3834 }
3835 // User has cancelled the menu (either by <esc> or clicking out of it),
3836 else
3837 {
3838 cancelled = true;
3839 }
3840 }
3841 else if( evt->Action() == TA_CHOICE_MENU_CLOSED )
3842 {
3843 break;
3844 }
3845 }
3846
3847 if( cancelled )
3848 return std::optional<int>();
3849 else
3850 return selectedNetCode;
3851 }
3852
3853 std::optional<int> findStitchedZoneNet( PCB_VIA* aVia )
3854 {
3855 const VECTOR2I position = aVia->GetPosition();
3856 PCB_DISPLAY_OPTIONS opts = m_frame->GetDisplayOptions();
3857 std::set<int> netcodeList;
3858
3859 // See if there are any connections available on a high-contrast layer
3860 if( opts.m_ContrastModeDisplay == HIGH_CONTRAST_MODE::DIMMED
3861 || opts.m_ContrastModeDisplay == HIGH_CONTRAST_MODE::HIDDEN )
3862 {
3863 if( aVia->GetLayerSet().test( m_frame->GetActiveLayer() ) )
3864 {
3865 for( ZONE* z : m_board->Zones() )
3866 {
3867 if( z->IsOnLayer( m_frame->GetActiveLayer() ) )
3868 {
3869 if( z->HitTestFilledArea( m_frame->GetActiveLayer(), position ) )
3870 netcodeList.insert( z->GetNetCode() );
3871 }
3872 }
3873 }
3874 }
3875
3876 // If there's only one, return it.
3877 if( netcodeList.size() == 1 )
3878 return *netcodeList.begin();
3879
3880 // See if there are any connections available on a visible layer
3881 LSET lset = LSET( m_board->GetVisibleLayers() & aVia->GetLayerSet() );
3882
3883 for( ZONE* z : m_board->Zones() )
3884 {
3885 for( PCB_LAYER_ID layer : lset.Seq() )
3886 {
3887 if( z->IsOnLayer( layer ) )
3888 {
3889 if( z->HitTestFilledArea( layer, position ) )
3890 netcodeList.insert( z->GetNetCode() );
3891 }
3892 }
3893 }
3894
3895 // If there's only one, return it.
3896 if( netcodeList.size() == 1 )
3897 return *netcodeList.begin();
3898
3899 if( netcodeList.size() > 1 )
3900 {
3901 // The net assignment is ambiguous. Let the user decide.
3902 return selectPossibleNetsByPopupMenu( netcodeList );
3903 }
3904 else
3905 {
3907 }
3908 }
3909
3910 void SnapItem( BOARD_ITEM *aItem ) override
3911 {
3912 m_gridHelper.SetSnap( !( m_modifiers & MD_SHIFT ) );
3913
3914 MAGNETIC_SETTINGS* settings = m_frame->GetMagneticItemsSettings();
3915 PCB_VIA* via = static_cast<PCB_VIA*>( aItem );
3916 VECTOR2I position = via->GetPosition();
3917
3918 if( settings->tracks != MAGNETIC_OPTIONS::NO_EFFECT && m_gridHelper.GetSnap() )
3919 {
3920 if( PCB_TRACK* track = findTrack( via ) )
3921 {
3922 SEG trackSeg( track->GetStart(), track->GetEnd() );
3923 VECTOR2I snap = m_gridHelper.AlignToSegment( position, trackSeg );
3924
3925 aItem->SetPosition( snap );
3926 return;
3927 }
3928 }
3929
3930 if( settings->pads != MAGNETIC_OPTIONS::NO_EFFECT && m_gridHelper.GetSnap() )
3931 {
3932 if( PAD* pad = findPad( via ) )
3933 {
3934 aItem->SetPosition( pad->GetPosition() );
3935 return;
3936 }
3937 }
3938
3939 if( settings->graphics && m_gridHelper.GetSnap() )
3940 {
3941 if( PCB_SHAPE* shape = findGraphic( via ) )
3942 {
3943 if( shape->IsAnyFill() )
3944 {
3945 aItem->SetPosition( shape->GetPosition() );
3946 }
3947 else
3948 {
3949 switch( shape->GetShape() )
3950 {
3951 case SHAPE_T::SEGMENT:
3952 {
3953 SEG seg( shape->GetStart(), shape->GetEnd() );
3954 VECTOR2I snap = m_gridHelper.AlignToSegment( position, seg );
3955 aItem->SetPosition( snap );
3956 break;
3957 }
3958
3959 case SHAPE_T::ARC:
3960 {
3961 if( ( shape->GetEnd() - position ).SquaredEuclideanNorm() <
3962 ( shape->GetStart() - position ).SquaredEuclideanNorm() )
3963 {
3964 aItem->SetPosition( shape->GetEnd() );
3965 }
3966 else
3967 {
3968 aItem->SetPosition( shape->GetStart() );
3969 }
3970
3971 break;
3972 }
3973
3974 case SHAPE_T::POLY:
3975 {
3976 if( !shape->IsPolyShapeValid() )
3977 {
3978 aItem->SetPosition( shape->GetPosition() );
3979 break;
3980 }
3981
3982 const SHAPE_POLY_SET& polySet = shape->GetPolyShape();
3983 std::optional<SEG> nearestSeg;
3984 int minDist = std::numeric_limits<int>::max();
3985
3986 for( int ii = 0; ii < polySet.OutlineCount(); ++ii )
3987 {
3988 const SHAPE_LINE_CHAIN& poly = polySet.Outline( ii );
3989
3990 for( int jj = 0; jj < poly.SegmentCount(); ++jj )
3991 {
3992 const SEG& seg = poly.GetSegment( jj );
3993 int dist = seg.Distance( position );
3994
3995 if( dist < minDist )
3996 {
3997 minDist = dist;
3998 nearestSeg = seg;
3999 }
4000 }
4001 }
4002
4003 if( nearestSeg )
4004 {
4005 VECTOR2I snap = m_gridHelper.AlignToSegment( position, *nearestSeg );
4006 aItem->SetPosition( snap );
4007 }
4008
4009 break;
4010 }
4011
4012 default:
4013 aItem->SetPosition( shape->GetPosition() );
4014 }
4015
4016 }
4017 }
4018
4019 }
4020 }
4021
4022 bool PlaceItem( BOARD_ITEM* aItem, BOARD_COMMIT& aCommit ) override
4023 {
4024 WX_INFOBAR* infobar = m_frame->GetInfoBar();
4025 PCB_VIA* via = static_cast<PCB_VIA*>( aItem );
4026 VECTOR2I viaPos = via->GetPosition();
4027 PCB_TRACK* track = findTrack( via );
4028 PAD* pad = findPad( via );
4029
4030 if( track )
4031 {
4032 via->SetNetCode( track->GetNetCode() );
4033 via->SetIsFree( false );
4034 }
4035 else if( pad )
4036 {
4037 via->SetNetCode( pad->GetNetCode() );
4038 via->SetIsFree( false );
4039 }
4040 else
4041 {
4042 std::optional<int> netcode = findStitchedZoneNet( via );
4043
4044 if( !netcode.has_value() ) // user cancelled net disambiguation menu
4045 return false;
4046
4047 via->SetNetCode( netcode.value() );
4048 via->SetIsFree( via->GetNetCode() > 0 );
4049 }
4050
4051 if( checkDRCViolation( via ) )
4052 {
4053 m_frame->ShowInfoBarError( _( "Via location violates DRC." ), true,
4055
4056 if( !m_allowDRCViolations )
4057 return false;
4058 }
4059 else
4060 {
4062 infobar->Dismiss();
4063 }
4064
4065 aCommit.Add( via );
4066
4067 // If the user explicitly disables snap (using shift), then don't break the tracks.
4068 // This will prevent PNS from being able to connect the via and track but
4069 // it is explicitly requested by the user
4070 if( track && m_gridHelper.GetSnap() )
4071 {
4072 VECTOR2I trackStart = track->GetStart();
4073 VECTOR2I trackEnd = track->GetEnd();
4074 SEG trackSeg( trackStart, trackEnd );
4075
4076 if( viaPos == trackStart || viaPos == trackEnd )
4077 return true;
4078
4079 if( !trackSeg.Contains( viaPos ) )
4080 return true;
4081
4082 aCommit.Modify( track );
4083 track->SetStart( trackStart );
4084 track->SetEnd( viaPos );
4085
4086 PCB_TRACK* newTrack = dynamic_cast<PCB_TRACK*>( track->Clone() );
4087 const_cast<KIID&>( newTrack->m_Uuid ) = KIID();
4088
4089 newTrack->SetStart( viaPos );
4090 newTrack->SetEnd( trackEnd );
4091 aCommit.Add( newTrack );
4092 }
4093
4094 return true;
4095 }
4096
4097 std::unique_ptr<BOARD_ITEM> CreateItem() override
4098 {
4099 BOARD_DESIGN_SETTINGS& bds = m_board->GetDesignSettings();
4100 PCB_VIA* via = new PCB_VIA( m_board );
4101
4102 via->SetNetCode( 0 );
4103 via->SetViaType( bds.m_CurrentViaType );
4104
4105 if( via->GetViaType() == VIATYPE::THROUGH )
4106 {
4107 via->SetLayerPair( B_Cu, F_Cu );
4108 }
4109 else
4110 {
4111 PCB_LAYER_ID first_layer = m_frame->GetActiveLayer();
4112 PCB_LAYER_ID last_layer;
4113
4114 // prepare switch to new active layer:
4115 if( first_layer != m_frame->GetScreen()->m_Route_Layer_TOP )
4116 last_layer = m_frame->GetScreen()->m_Route_Layer_TOP;
4117 else
4118 last_layer = m_frame->GetScreen()->m_Route_Layer_BOTTOM;
4119
4120 via->SetLayerPair( first_layer, last_layer );
4121 }
4122
4123 if( via->GetViaType() == VIATYPE::MICROVIA )
4124 {
4125 via->SetWidth( PADSTACK::ALL_LAYERS,
4126 via->GetEffectiveNetClass()->GetuViaDiameter() );
4127 via->SetDrill( via->GetEffectiveNetClass()->GetuViaDrill() );
4128 }
4129 else
4130 {
4131 via->SetWidth( PADSTACK::ALL_LAYERS, bds.GetCurrentViaSize() );
4132 via->SetDrill( bds.GetCurrentViaDrill() );
4133 }
4134
4135 return std::unique_ptr<BOARD_ITEM>( via );
4136 }
4137 };
4138
4139 VIA_PLACER placer( frame() );
4140
4141 SCOPED_DRAW_MODE scopedDrawMode( m_mode, MODE::VIA );
4142
4143 doInteractiveItemPlacement( aEvent, &placer, _( "Place via" ), IPO_REPEAT | IPO_SINGLE_CLICK );
4144
4145 return 0;
4146}
4147
4148
4149const unsigned int DRAWING_TOOL::WIDTH_STEP = pcbIUScale.mmToIU( 0.1 );
4150
4151
4153{
4154 // clang-format off
4179
4181
4185 // clang-format on
4186}
constexpr int ARC_HIGH_DEF
Definition: base_units.h:120
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
static TOOL_ACTION cancelInteractive
Definition: actions.h:72
static TOOL_ACTION selectItem
Select an item (specified as the event parameter).
Definition: actions.h:223
static TOOL_ACTION selectionCursor
Select a single item under the cursor position.
Definition: actions.h:217
static TOOL_ACTION updateUnits
Definition: actions.h:207
static TOOL_ACTION undo
Definition: actions.h:75
static TOOL_ACTION activatePointEditor
Definition: actions.h:265
static TOOL_ACTION doDelete
Definition: actions.h:85
static TOOL_ACTION cursorClick
Definition: actions.h:180
static TOOL_ACTION redo
Definition: actions.h:76
static TOOL_ACTION selectionClear
Clear the current selection.
Definition: actions.h:220
static TOOL_ACTION refreshPreview
Definition: actions.h:156
static TOOL_ACTION selectItems
Select a list of items (specified as the event parameter)
Definition: actions.h:228
static TOOL_ACTION resetLocalCoords
Definition: actions.h:210
Define the structure of a menu based on ACTIONs.
Definition: action_menu.h:49
TOOL_MANAGER * getToolManager() const
Return an instance of TOOL_MANAGER class.
void DisplayTitle(bool aDisplay=true)
Decide whether a title for a pop up menu should be displayed.
void Clear()
Remove all the entries from the menu (as well as its title).
void SetTitle(const wxString &aTitle) override
Set title for the menu.
Definition: action_menu.cpp:92
void SetIcon(BITMAPS aIcon)
Assign an icon for the entry.
Definition: action_menu.cpp:78
wxMenuItem * Add(const wxString &aLabel, int aId, BITMAPS aIcon)
Add a wxWidgets-style entry to the menu.
void SetLayerVisible(int aLayer, bool isVisible)
VECTOR2D m_LocalOrigin
Relative Screen cursor coordinate (on grid) in user units.
Definition: base_screen.h:90
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.
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
Container for design settings for a BOARD object.
DIM_PRECISION m_DimensionPrecision
Number of digits after the decimal.
void UseCustomTrackViaSize(bool aEnabled)
Enables/disables custom track/via size settings.
VIATYPE m_CurrentViaType
(VIA_BLIND_BURIED, VIA_THROUGH, VIA_MICROVIA)
DIM_UNITS_FORMAT m_DimensionUnitsFormat
bool GetTextUpright(PCB_LAYER_ID aLayer) const
int GetTextThickness(PCB_LAYER_ID aLayer) const
Return the default text thickness from the layer class for the given layer.
bool GetTextItalic(PCB_LAYER_ID aLayer) const
std::shared_ptr< DRC_ENGINE > m_DRCEngine
void SetViaSizeIndex(unsigned aIndex)
Set the current via size list index to aIndex.
int GetDRCEpsilon() const
Return an epsilon which accounts for rounding errors, etc.
unsigned GetViaSizeIndex() const
VECTOR2I GetTextSize(PCB_LAYER_ID aLayer) const
Return the default text size from the layer class for the given layer.
int GetLineThickness(PCB_LAYER_ID aLayer) const
Return the default graphic segment thickness from the layer class for the given layer.
DIM_TEXT_POSITION m_DimensionTextPosition
DIM_UNITS_MODE m_DimensionUnitsMode
std::vector< VIA_DIMENSION > m_ViasDimensionsList
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 void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
Definition: board_item.h:280
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 std::shared_ptr< SHAPE_SEGMENT > GetEffectiveHoleShape() const
Definition: board_item.cpp:289
virtual bool HasHole() const
Definition: board_item.h:153
const FOOTPRINTS & Footprints() const
Definition: board.h:338
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
constexpr BOX2< Vec > & Inflate(coord_type dx, coord_type dy)
Inflates the rectangle horizontally by dx and vertically by dy.
Definition: box2.h:558
constexpr size_type GetWidth() const
Definition: box2.h:214
constexpr Vec Centre() const
Definition: box2.h:97
constexpr size_type GetHeight() const
Definition: box2.h:215
constexpr coord_type GetLeft() const
Definition: box2.h:228
constexpr bool Contains(const Vec &aPoint) const
Definition: box2.h:168
constexpr coord_type GetRight() const
Definition: box2.h:217
constexpr coord_type GetTop() const
Definition: box2.h:229
constexpr coord_type GetBottom() const
Definition: box2.h:222
COMMIT & Modify(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Modify a given item in the model.
Definition: commit.h:108
COMMIT & Add(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Add a new item to the model.
Definition: commit.h:80
void AddItem(const TOOL_ACTION &aAction, const SELECTION_CONDITION &aCondition, int aOrder=ANY_ORDER)
Add a menu entry to run a TOOL_ACTION on selected items.
void AddSeparator(int aOrder=ANY_ORDER)
Add a separator to the menu.
void AddCheckItem(const TOOL_ACTION &aAction, const SELECTION_CONDITION &aCondition, int aOrder=ANY_ORDER)
Add a checked menu entry to run a TOOL_ACTION on selected items.
void AddMenu(ACTION_MENU *aMenu, const SELECTION_CONDITION &aCondition=SELECTION_CONDITIONS::ShowAlways, int aOrder=ANY_ORDER)
Add a submenu to the menu.
std::list< std::unique_ptr< EDA_ITEM > > & GetImportedItems()
int ShowQuasiModal()
int ShowModal() override
Implementing DIALOG_TRACK_VIA_SIZE_BASE.
Tool responsible for drawing graphical elements like lines, arcs, circles, etc.
Definition: drawing_tool.h:55
TEXT_ATTRIBUTES m_textAttrs
Definition: drawing_tool.h:402
MODE GetDrawingMode() const
Return the current drawing mode of the DRAWING_TOOL or MODE::NONE if not currently in any drawing mod...
PCB_SELECTION m_preview
Definition: drawing_tool.h:404
PCB_LAYER_ID m_layer
Definition: drawing_tool.h:400
int DrawTable(const TOOL_EVENT &aEvent)
int SetAnchor(const TOOL_EVENT &aEvent)
Place the footprint anchor (only in footprint editor).
int DrawDimension(const TOOL_EVENT &aEvent)
Start interactively drawing a dimension.
int DrawVia(const TOOL_EVENT &aEvent)
KIGFX::VIEW_CONTROLS * m_controls
Definition: drawing_tool.h:394
KIGFX::VIEW * m_view
Definition: drawing_tool.h:393
STROKE_PARAMS m_stroke
Definition: drawing_tool.h:401
int DrawZone(const TOOL_EVENT &aEvent)
Start interactively drawing a zone.
int DrawBezier(const TOOL_EVENT &aEvent)
Start interactively drawing a bezier curve.
int DrawLine(const TOOL_EVENT &aEvent)
Start interactively drawing a line.
int DrawArc(const TOOL_EVENT &aEvent)
Start interactively drawing an arc.
VECTOR2I getClampedDifferenceEnd(const VECTOR2I &aOrigin, const VECTOR2I &aEnd)
Clamps the end vector to respect numeric limits of difference representation.
Definition: drawing_tool.h:341
int ToggleHV45Mode(const TOOL_EVENT &toolEvent)
Toggle the horizontal/vertical/45-degree constraint for drawing tools.
int PlaceCharacteristics(const TOOL_EVENT &aEvent)
bool drawShape(const TOOL_EVENT &aTool, PCB_SHAPE **aGraphic, std::optional< VECTOR2D > aStartingPoint, std::stack< PCB_SHAPE * > *aCommittedGraphics)
Start drawing a selected shape (i.e.
void setTransitions() override
This method is meant to be overridden in order to specify handlers for events.
int PlaceImportedGraphics(const TOOL_EVENT &aEvent)
Place a drawing imported from a DXF or SVG file in footprint editor.
int PlaceStackup(const TOOL_EVENT &aEvent)
VECTOR2I getClampedRadiusEnd(const VECTOR2I &aOrigin, const VECTOR2I &aEnd)
Clamps the end vector to respect numeric limits of radius representation.
Definition: drawing_tool.h:371
bool m_inDrawingTool
Definition: drawing_tool.h:398
static const unsigned int WIDTH_STEP
Definition: drawing_tool.h:408
static const unsigned int COORDS_PADDING
Definition: drawing_tool.h:409
bool drawArc(const TOOL_EVENT &aTool, PCB_SHAPE **aGraphic, std::optional< VECTOR2D > aStartingPoint)
Start drawing an arc.
void Reset(RESET_REASON aReason) override
Bring the tool to a known, initial state.
int DrawCircle(const TOOL_EVENT &aEvent)
Start interactively drawing a circle.
int PlaceTuningPattern(const TOOL_EVENT &aEvent)
BOARD * m_board
Definition: drawing_tool.h:395
int DrawRectangle(const TOOL_EVENT &aEvent)
Start interactively drawing a rectangle.
bool Init() override
Init() is called once upon a registration of the tool.
void constrainDimension(PCB_DIMENSION_BASE *aDim)
Force the dimension lime to be drawn on multiple of 45 degrees.
int PlaceText(const TOOL_EVENT &aEvent)
Display a dialog that allows one to input text and its settings and then lets the user decide where t...
PCB_BASE_EDIT_FRAME * m_frame
Definition: drawing_tool.h:396
void UpdateStatusBar() const
int PlaceReferenceImage(const TOOL_EVENT &aEvent)
Display a dialog that allows one to select a reference image and then decide where to place the image...
bool getSourceZoneForAction(ZONE_MODE aMode, ZONE **aZone)
Draw a polygon, that is added as a zone or a keepout area.
std::unique_ptr< PCB_SHAPE > drawOneBezier(const TOOL_EVENT &aTool, const OPT_VECTOR2I &aStartingPoint, const OPT_VECTOR2I &aStartingControl1Point, DRAW_ONE_RESULT &aResult)
Draw a bezier curve.
int m_DisallowFlags
Definition: drc_rule.h:191
SEVERITY GetSeverity() const
Definition: drc_rule.h:173
const MINOPTMAX< int > & GetValue() const
Definition: drc_rule.h:152
void ShowInfoBarMsg(const wxString &aMsg, bool aShowCloseButton=false)
Show the WX_INFOBAR displayed on the top of the canvas with a message and an info icon on the left of...
virtual PICKED_ITEMS_LIST * PopCommandFromUndoList()
Return the last command to undo and remove it from list, nothing is deleted.
void ShowInfoBarError(const wxString &aErrorMsg, bool aShowCloseButton=false, WX_INFOBAR::MESSAGE_TYPE aType=WX_INFOBAR::MESSAGE_TYPE::GENERIC)
Show the WX_INFOBAR displayed on the top of the canvas with a message and an error icon on the left o...
bool IsType(FRAME_T aType) const
WX_INFOBAR * GetInfoBar()
void SetMsgPanel(const std::vector< MSG_PANEL_ITEM > &aList)
Clear the message panel and populates it with the contents of aList.
void DisplayConstraintsMsg(const wxString &msg)
void SetCurrentCursor(KICURSOR aCursor)
Set the current cursor shape for this panel.
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
virtual void SetPosition(const VECTOR2I &aPos)
Definition: eda_item.h:253
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition: eda_item.h:135
const KIID m_Uuid
Definition: eda_item.h:498
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
const VECTOR2I & GetBezierC2() const
Definition: eda_shape.h:258
void SetBezierC2(const VECTOR2I &aPt)
Definition: eda_shape.h:257
void SetCenter(const VECTOR2I &aCenter)
Definition: eda_shape.cpp:912
virtual void SetFilled(bool aFlag)
Definition: eda_shape.h:136
void RebuildBezierToSegmentsPointsList(int aMaxError)
Rebuild the m_bezierPoints vertex list that approximate the Bezier curve by a list of segments.
Definition: eda_shape.cpp:859
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
Definition: eda_shape.h:215
void SetStart(const VECTOR2I &aStart)
Definition: eda_shape.h:177
void SetShape(SHAPE_T aShape)
Definition: eda_shape.h:167
void SetEnd(const VECTOR2I &aEnd)
Definition: eda_shape.h:219
void SetBezierC1(const VECTOR2I &aPt)
Definition: eda_shape.h:254
double GetLength() const
Definition: eda_shape.cpp:374
void SetArcAngleAndEnd(const EDA_ANGLE &aAngle, bool aCheckNegativeAngle=false)
Set the end point from the angle center and start.
Definition: eda_shape.cpp:1061
void SetTextSize(VECTOR2I aNewSize, bool aEnforceMinTextSize=true)
Definition: eda_text.cpp:526
virtual bool IsVisible() const
Definition: eda_text.h:174
void SetTextPos(const VECTOR2I &aPoint)
Definition: eda_text.cpp:571
void SetMirrored(bool isMirrored)
Definition: eda_text.cpp:386
void SetTextThickness(int aWidth)
The TextThickness is that set by the user.
Definition: eda_text.cpp:284
void SetItalic(bool aItalic)
Set the text to be italic - this will also update the font if needed.
Definition: eda_text.cpp:300
void MoveAnchorPosition(const VECTOR2I &aMoveVector)
Move the reference point of the footprint.
Definition: footprint.cpp:2472
VECTOR2I GetPosition() const override
Definition: footprint.h:229
bool GetSnap() const
Definition: grid_helper.h:112
void SetSnap(bool aSnap)
Definition: grid_helper.h:111
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
virtual RENDER_SETTINGS * GetSettings()=0
Return a pointer to current settings that are going to be used when drawing items.
Represents an assistant draw when interactively drawing an arc on a canvas.
Definition: arc_assistant.h:41
void SetUnits(EDA_UNITS aUnits)
Definition: arc_assistant.h:71
Manage the construction of a circular arc though sequential setting of critical points: center,...
VECTOR2I GetOrigin() const
< Get the center point of the arc (valid when state > SET_ORIGIN)
ARC_STEPS GetStep() const
Get the current step the manager is on (useful when drawing something depends on the current state)
void ToggleClockwise()
Set angle snapping (for the next point)
VECTOR2I GetStartRadiusEnd() const
Get the coordinates of the arc end point.
VECTOR2I GetEndRadiusEnd() const
Get the radius of the arc (valid if step >= SET_START)
@ SET_ANGLE
Waiting to lock in the arc end point.
@ SET_START
Waiting to lock in the arc start point.
Represents an assistant draw when interactively drawing a bezier on a canvas.
void SetUnits(EDA_UNITS aUnits)
Manage the construction of a bezier through a series of steps.
VECTOR2I GetControlC2() const
Get the coordinates of the arc end point.
VECTOR2I GetStart() const
< Get the center point of the arc (valid when state > SET_ORIGIN)
@ SET_END
Waiting to lock in the end point.
@ SET_START
Waiting to lock in the start point.
BEZIER_STEPS GetStep() const
Get the current step the manager is on (useful when drawing something depends on the current state)
void AddPoint(const VECTOR2I &aPt, bool aLockIn)
Add a point to the construction manager.
void RemoveLastPoint()
Undo the last point, and move the manager back to the previous step.
Represents an assistant draw when interactively drawing a line or circle on a canvas.
Represent a very simple geometry manager for items that have a start and end point.
void SetOrigin(const VECTOR2I &aOrigin)
< Set the origin of the ruler (the fixed end)
void Reset()
Reset the manager to the initial state.
void SetEnd(const VECTOR2I &aEnd)
Set the current end of the rectangle (the end that moves with the cursor.
Container for all the knowledge about how graphical objects are drawn on any output surface/device.
const std::set< int > & GetHighlightNetCodes() const
Return the netcode of currently highlighted net.
virtual void CaptureCursor(bool aEnabled)
Force the cursor to stay within the drawing panel area.
virtual void ForceCursorPosition(bool aEnabled, const VECTOR2D &aPosition=VECTOR2D(0, 0))
Place the cursor immediately at a given point.
virtual void ShowCursor(bool aEnabled)
Enable or disables display of cursor.
virtual void WarpMouseCursor(const VECTOR2D &aPosition, bool aWorldCoordinates=false, bool aWarpView=false)=0
If enabled (.
VECTOR2D GetCursorPosition() const
Return the current cursor position in world coordinates.
virtual VECTOR2D GetMousePosition(bool aWorldCoordinates=true) const =0
Return the current mouse pointer position.
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.
virtual void SetAutoPan(bool aEnabled)
Turn on/off auto panning (this feature is used when there is a tool active (eg.
virtual void PinCursorInsideNonAutoscrollArea(bool aWarpMouseCursor)=0
bool IsBOARD_ITEM() const
Definition: view_item.h:102
void ShowPreview(bool aShow=true)
Definition: view.cpp:1731
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
int Query(const BOX2I &aRect, std::vector< LAYER_ITEM_PAIR > &aResult) const
Find all visible items that touch or are within the rectangle aRect.
Definition: view.cpp:410
virtual void Update(const VIEW_ITEM *aItem, int aUpdateFlags) const
For dynamic VIEWs, inform the associated VIEW that the graphical representation of this item has chan...
Definition: view.cpp:1673
void ClearPreview()
Definition: view.cpp:1695
void RecacheAllItems()
Rebuild GAL display lists.
Definition: view.cpp:1439
std::pair< VIEW_ITEM *, int > LAYER_ITEM_PAIR
Definition: view.h:71
void AddToPreview(VIEW_ITEM *aItem, bool aTakeOwnership=true)
Definition: view.cpp:1717
bool IsLayerVisible(int aLayer) const
Return information about visibility of a particular layer.
Definition: view.h:418
PAINTER * GetPainter() const
Return the painter object used by the view for drawing #VIEW_ITEMS.
Definition: view.h:216
Definition: kiid.h:49
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
static LSET AllLayersMask()
Definition: lset.cpp:601
static LSET AllCuMask(int aCuLayerCount=MAX_CU_LAYERS)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Definition: lset.cpp:572
LSEQ Seq(const LSEQ &aSequence) const
Return an LSEQ from the union of this LSET and a desired sequence.
Definition: lset.cpp:297
T Min() const
Definition: minoptmax.h:33
const wxString & GetNetname() const
Definition: netinfo.h:114
Container for NETINFO_ITEM elements, which are the nets.
Definition: netinfo.h:346
static const int ORPHANED
Constant that forces initialization of a netinfo item to the NETINFO_ITEM ORPHANED (typically -1) whe...
Definition: netinfo.h:385
NETINFO_ITEM * GetNetItem(int aNetCode) const
void ForEachUniqueLayer(const std::function< void(PCB_LAYER_ID)> &aMethod) const
Runs the given callable for each active unique copper layer in this padstack, meaning F_Cu for MODE::...
Definition: padstack.cpp:881
static constexpr PCB_LAYER_ID ALL_LAYERS
! Temporary layer identifier to identify code that is not padstack-aware
Definition: padstack.h:144
Definition: pad.h:54
static TOOL_ACTION deleteLastPoint
Definition: pcb_actions.h:214
static TOOL_ACTION toggleHV45Mode
Definition: pcb_actions.h:520
static TOOL_ACTION drawRuleArea
Definition: pcb_actions.h:206
static TOOL_ACTION changeDimensionArrows
Switch between dimension arrow directions.
Definition: pcb_actions.h:227
static TOOL_ACTION drawBezier
Definition: pcb_actions.h:189
static TOOL_ACTION placeText
Definition: pcb_actions.h:191
static TOOL_ACTION drawOrthogonalDimension
Definition: pcb_actions.h:202
static TOOL_ACTION drawRectangle
Definition: pcb_actions.h:186
static TOOL_ACTION setAnchor
Definition: pcb_actions.h:213
static TOOL_ACTION placeReferenceImage
Definition: pcb_actions.h:190
static TOOL_ACTION drawCircle
Definition: pcb_actions.h:187
static TOOL_ACTION tuneDiffPair
Definition: pcb_actions.h:243
static TOOL_ACTION trackViaSizeChanged
Definition: pcb_actions.h:382
static TOOL_ACTION layerChanged
Definition: pcb_actions.h:370
static TOOL_ACTION drawTable
Definition: pcb_actions.h:193
static TOOL_ACTION drawTextBox
Definition: pcb_actions.h:192
static TOOL_ACTION drawZoneCutout
Definition: pcb_actions.h:207
static TOOL_ACTION drawPolygon
Definition: pcb_actions.h:185
static TOOL_ACTION drawRadialDimension
Definition: pcb_actions.h:201
static TOOL_ACTION tuneSingleTrack
Definition: pcb_actions.h:242
static TOOL_ACTION properties
Activation of the edit tool.
Definition: pcb_actions.h:163
static TOOL_ACTION drawLeader
Definition: pcb_actions.h:203
static TOOL_ACTION placeCharacteristics
Definition: pcb_actions.h:209
static TOOL_ACTION tuneSkew
Definition: pcb_actions.h:244
static TOOL_ACTION incWidth
Increase width of currently drawn line.
Definition: pcb_actions.h:218
static TOOL_ACTION clearHighlight
Definition: pcb_actions.h:546
static TOOL_ACTION spacingDecrease
Definition: pcb_actions.h:195
static TOOL_ACTION placeImportedGraphics
Definition: pcb_actions.h:212
static TOOL_ACTION drawVia
Definition: pcb_actions.h:205
static TOOL_ACTION drawArc
Definition: pcb_actions.h:188
static TOOL_ACTION drawSimilarZone
Definition: pcb_actions.h:208
static TOOL_ACTION decWidth
Decrease width of currently drawn line.
Definition: pcb_actions.h:221
static TOOL_ACTION drawCenterDimension
Definition: pcb_actions.h:200
static TOOL_ACTION arcPosture
Switch posture when drawing arc.
Definition: pcb_actions.h:224
static TOOL_ACTION closeOutline
Definition: pcb_actions.h:215
static TOOL_ACTION amplIncrease
Definition: pcb_actions.h:196
static TOOL_ACTION amplDecrease
Definition: pcb_actions.h:197
static TOOL_ACTION lengthTunerSettings
Definition: pcb_actions.h:198
static TOOL_ACTION spacingIncrease
Definition: pcb_actions.h:194
static TOOL_ACTION drawLine
Definition: pcb_actions.h:184
static TOOL_ACTION placeStackup
Definition: pcb_actions.h:210
static TOOL_ACTION drawAlignedDimension
Definition: pcb_actions.h:199
static TOOL_ACTION drawZone
Definition: pcb_actions.h:204
virtual bool HitTest(const VECTOR2I &aPosition, int aAccuracy=0) const override
Test if aPosition is inside or on the boundary of this item.
Definition: pcb_track.cpp:1874
Common, abstract interface for edit frames.
wxString GetDesignRulesPath()
Return the absolute path to the design rules file for the currently-loaded board.
void ClearListAndDeleteItems(PICKED_ITEMS_LIST *aList)
Definition: undo_redo.cpp:683
int ShowTextBoxPropertiesDialog(PCB_TEXTBOX *aTextBox)
APPEARANCE_CONTROLS * GetAppearancePanel()
void PutDataInPreviousState(PICKED_ITEMS_LIST *aList)
Used in undo or redo command.
Definition: undo_redo.cpp:259
void SetObjectVisible(GAL_LAYER_ID aLayer, bool aVisible=true)
const PCB_DISPLAY_OPTIONS & GetDisplayOptions() const
Display options control the way tracks, vias, outlines and other things are shown (for instance solid...
virtual PCB_LAYER_ID GetActiveLayer() const
virtual MAGNETIC_SETTINGS * GetMagneticItemsSettings()
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.
virtual BOARD_ITEM_CONTAINER * GetModel() const =0
Abstract dimension API.
void Update()
Update the dimension's cached text and geometry.
int GetLineThickness() const
void SetExtensionOffset(int aOffset)
void SetLineThickness(int aWidth)
void SetArrowLength(int aLength)
virtual const VECTOR2I & GetStart() const
The dimension's origin is the first feature point for the dimension.
DIM_ARROW_DIRECTION GetArrowDirection() const
virtual void SetEnd(const VECTOR2I &aPoint)
virtual void SetStart(const VECTOR2I &aPoint)
void SetArrowDirection(const DIM_ARROW_DIRECTION &aDirection)
int GetArrowLength() const
virtual const VECTOR2I & GetEnd() const
For better understanding of the points that make a dimension:
double GetAngle() const
Return the angle of the crossbar.
void SetHeight(int aHeight)
Set the distance from the feature points to the crossbar line.
Mark the center of a circle or arc with a cross shape.
A leader is a dimension-like object pointing to a specific point.
An orthogonal dimension is like an aligned dimension, but the extension lines are locked to the X or ...
A radial dimension indicates either the radius or diameter of an arc or circle.
VECTOR2I GetKnee() const
HIGH_CONTRAST_MODE m_ContrastModeDisplay
How inactive layers are displayed.
virtual KIGFX::PCB_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
The main frame for Pcbnew.
VECTOR2I AlignToSegment(const VECTOR2I &aPoint, const SEG &aSeg)
A set of BOARD_ITEMs (i.e., without duplicates).
Definition: pcb_group.h:53
Object to handle a bitmap image that can be inserted in a PCB.
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:
PCB_SELECTION & GetSelection()
EDA_ITEM * GetTopLeftItem(bool aFootprintsOnly=false) const override
bool HitTest(const VECTOR2I &aPosition, int aAccuracy=0) const override
Test if aPosition is inside or on the boundary of this item.
Definition: pcb_shape.h:123
void SetPosition(const VECTOR2I &aPos) override
Definition: pcb_shape.h:78
void SetLayer(PCB_LAYER_ID aLayer) override
Set the layer this item is on.
Definition: pcb_shape.cpp:179
void SetStroke(const STROKE_PARAMS &aStroke) override
Definition: pcb_shape.h:92
void Normalize() override
Perform any normalization required after a user rotate and/or flip.
Definition: pcb_shape.cpp:473
VECTOR2I GetPosition() const override
Definition: pcb_shape.h:79
T * frame() const
virtual bool Is45Limited() const
Should the tool use its 45° mode option?
KIGFX::VIEW_CONTROLS * controls() const
BOARD * board() const
@ IPO_SINGLE_CLICK
Create an item immediately on placement starting, otherwise show the pencil cursor until the item is ...
@ IPO_REPEAT
Allow repeat placement of the item.
bool m_isFootprintEditor
void doInteractiveItemPlacement(const TOOL_EVENT &aTool, INTERACTIVE_PLACER_BASE *aPlacer, const wxString &aCommitMessage, int aOptions=IPO_ROTATE|IPO_FLIP|IPO_REPEAT)
Helper function for performing a common interactive idiom: wait for a left click, place an item there...
const PCB_SELECTION & selection() const
FOOTPRINT * footprint() const
void SetEnd(const VECTOR2I &aEnd)
Definition: pcb_track.h:148
void SetStart(const VECTOR2I &aStart)
Definition: pcb_track.h:151
virtual EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
Definition: pcb_track.cpp:68
const VECTOR2I & GetStart() const
Definition: pcb_track.h:152
const VECTOR2I & GetEnd() const
Definition: pcb_track.h:149
virtual int GetWidth() const
Definition: pcb_track.h:146
VECTOR2I GetPosition() const override
Definition: pcb_track.h:543
std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer=UNDEFINED_LAYER, FLASHING aFlash=FLASHING::DEFAULT) const override
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
Definition: pcb_track.cpp:2109
const PADSTACK & Padstack() const
Definition: pcb_track.h:446
int GetWidth() const override
Definition: pcb_track.cpp:380
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition: pcb_track.cpp:1222
const BOX2I GetBoundingBox() const override
Return the orthogonal bounding box of this object for display purposes.
Definition: pcb_track.cpp:726
virtual COMMON_SETTINGS * GetCommonSettings() const
Definition: pgm_base.cpp:687
virtual SETTINGS_MANAGER & GetSettingsManager() const
Definition: pgm_base.h:125
A holder to handle information on schematic or board items.
void PushItem(const ITEM_PICKER &aItem)
Push aItem to the top of the list.
unsigned GetCount() const
ROUTING_SETTINGS & Settings()
Definition: pns_router.h:199
ROUTER * Router() const
Class that handles the drawing of a polygon, including management of last corner deletion and drawing...
bool AddPoint(const VECTOR2I &aPt)
Lock in a polygon point.
@ DIRECT
Unconstrained point-to-point.
void SetCursorPosition(const VECTOR2I &aPos)
Set the current cursor position.
bool NewPointClosesOutline(const VECTOR2I &aPt) const
std::optional< VECTOR2I > DeleteLastCorner()
Remove the last-added point from the polygon.
void SetFinished()
Mark the polygon finished and update the client.
void SetLeaderMode(LEADER_MODE aMode)
Set the leader mode to use when calculating the leader/returner lines.
void Reset()
Clear the manager state and start again.
RAII class that sets an value at construction and resets it to the original value at destruction.
Definition: seg.h:42
int Distance(const SEG &aSeg) const
Compute minimum Euclidean distance to segment aSeg.
Definition: seg.cpp:388
bool Contains(const SEG &aSeg) const
Definition: seg.h:314
int AddItemToSel(const TOOL_EVENT &aEvent)
virtual void Add(EDA_ITEM *aItem)
Definition: selection.cpp:42
virtual void Remove(EDA_ITEM *aItem)
Definition: selection.cpp:60
virtual void Clear() override
Remove all the stored items from the group.
Definition: selection.h:93
int Size() const
Returns the number of selected parts.
Definition: selection.h:116
void SetReferencePoint(const VECTOR2I &aP)
Definition: selection.cpp:178
bool Empty() const
Checks if there is anything selected.
Definition: selection.h:110
T * GetAppSettings(const wxString &aFilename)
Return a handle to the a given settings by type.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
virtual const SEG GetSegment(int aIndex) const override
int SegmentCount() const
Return the number of segments in this line chain.
Represent a set of closed polygons.
bool Collide(const SHAPE *aShape, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const override
Check if the boundary of shape (this) lies closer to the shape aShape than aClearance,...
SHAPE_LINE_CHAIN & Outline(int aIndex)
Return the reference to aIndex-th outline in the set.
int OutlineCount() const
Return the number of outlines in the set.
int GetWidth() const
void SetLineStyle(LINE_STYLE aLineStyle)
void SetWidth(int aWidth)
void SetColor(const KIGFX::COLOR4D &aColor)
GR_TEXT_H_ALIGN_T m_Halign
GR_TEXT_V_ALIGN_T m_Valign
virtual void PopTool(const TOOL_EVENT &aEvent)
Pops a tool from the stack.
virtual void PushTool(const TOOL_EVENT &aEvent)
NB: the definition of "tool" is different at the user level.
TOOL_MANAGER * GetToolManager() const
Return the MVC controller.
Definition: tools_holder.h:55
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
@ SHUTDOWN
Tool is being shut down.
Definition: tool_base.h:84
Generic, UI-independent tool event.
Definition: tool_event.h:168
bool HasPosition() const
Returns if it this event has a valid position (true for mouse events and context-menu or hotkey-based...
Definition: tool_event.h:257
bool DisableGridSnapping() const
Definition: tool_event.h:368
const VECTOR2D Position() const
Return mouse cursor position in world coordinates.
Definition: tool_event.h:290
bool IsReactivate() const
Control whether the tool is first being pushed to the stack or being reactivated after a pause.
Definition: tool_event.h:270
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 SetContextMenu(ACTION_MENU *aMenu, CONTEXT_MENU_TRIGGER aTrigger=CMENU_BUTTON)
Assign a context menu and tells when it should be activated.
void RunMainStack(std::function< void()> aFunc)
Call a function using the main stack.
void Go(int(T::*aStateFunc)(const TOOL_EVENT &), const TOOL_EVENT_LIST &aConditions=TOOL_EVENT(TC_ANY, TA_ANY))
Define which state (aStateFunc) to go when a certain event arrives (aConditions).
std::unique_ptr< TOOL_MENU > m_menu
The functions below are not yet implemented - their interface may change.
TOOL_EVENT * Wait(const TOOL_EVENT_LIST &aEventList=TOOL_EVENT(TC_ANY, TA_ANY))
Suspend execution of the tool until an event specified in aEventList arrives.
void Activate()
Run the tool.
void DeactivateTool()
Deactivate the currently active tool.
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
void PrimeTool(const VECTOR2D &aPosition)
"Prime" a tool by sending a cursor left-click event with the mouse position set to the passed in posi...
TOOLS_HOLDER * GetToolHolder() const
Definition: tool_manager.h:406
bool PostAction(const std::string &aActionName, T aParam)
Run the specified action after the current action (coroutine) ends.
Definition: tool_manager.h:235
void VetoContextMenuMouseWarp()
Disable mouse warping after the current context menu is closed.
Definition: tool_manager.h:515
KIGFX::VIEW * GetView() const
Definition: tool_manager.h:395
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
EDA_UNITS GetUserUnits() const
void update() override
Update menu state stub.
OPT_TOOL_EVENT eventHandler(const wxMenuEvent &aEvent) override
Event handler stub.
ACTION_MENU * create() const override
Return an instance of this class. It has to be overridden in inheriting classes.
A modified version of the wxInfoBar class that allows us to:
Definition: wx_infobar.h:76
void Dismiss() override
Dismisses the infobar and updates the containing layout and AUI manager (if one is provided).
Definition: wx_infobar.cpp:190
MESSAGE_TYPE GetMessageType() const
Definition: wx_infobar.h:101
An adjunct helper to the DRAWING_TOOL interactive tool, which handles incoming geometry changes from ...
void OnGeometryChange(const POLYGON_GEOM_MANAGER &aMgr) override
Called when the polygon is complete.
ZONE * GetZone() const
static bool IsZoneFillAction(const TOOL_EVENT *aEvent)
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
bool GetIsRuleArea() const
Accessors to parameters used in Rule Area zones:
Definition: zone.h:752
bool GetDoNotAllowVias() const
Definition: zone.h:769
SHAPE_POLY_SET * Outline()
Definition: zone.h:368
virtual bool IsOnLayer(PCB_LAYER_ID) const override
Test to see if this object is on the given layer.
Definition: zone.cpp:642
PCB_LAYER_ID GetFirstLayer() const
Definition: zone.cpp:499
This file is part of the common library.
#define TOGGLE(a)
static void updateArcFromConstructionMgr(const KIGFX::PREVIEW::ARC_GEOM_MANAGER &aMgr, PCB_SHAPE &aArc)
Update an arc PCB_SHAPE from the current state of an Arc Geometry Manager.
static void updateSegmentFromGeometryMgr(const KIGFX::PREVIEW::TWO_POINT_GEOMETRY_MANAGER &aMgr, PCB_SHAPE *aGraphic)
Update a PCB_SHAPE from the current state of a #TWO_POINT_GEOMETRY_MANAGER.
static void updateBezierFromConstructionMgr(const KIGFX::PREVIEW::BEZIER_GEOM_MANAGER &aMgr, PCB_SHAPE &aBezier)
Update a bezier PCB_SHAPE from the current state of a Bezier Geometry Manager.
@ DISALLOW_CONSTRAINT
Definition: drc_rule.h:66
@ CLEARANCE_CONSTRAINT
Definition: drc_rule.h:49
@ HOLE_CLEARANCE_CONSTRAINT
Definition: drc_rule.h:51
#define _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
static constexpr EDA_ANGLE ANGLE_90
Definition: eda_angle.h:403
std::vector< EDA_ITEM * > EDA_ITEMS
Define list of drawing items for screens.
Definition: eda_item.h:548
#define IS_NEW
New item, just created.
#define IS_MOVING
Item being moved.
SHAPE_T
Definition: eda_shape.h:43
@ NONE
Definition: eda_shape.h:69
EDA_UNITS
Definition: eda_units.h:46
void ConnectBoardShapes(std::vector< PCB_SHAPE * > &aShapeList, std::vector< std::unique_ptr< PCB_SHAPE > > &aNewShapes, int aChainingEpsilon)
Connects shapes to each other, making continious contours (adjacent shapes will have a common vertex)...
@ FRAME_PCB_EDITOR
Definition: frame_type.h:42
a few functions useful in geometry calculations.
VECTOR2< T > GetVectorSnapped45(const VECTOR2< T > &aVec, bool only45=false)
Snap a vector onto the nearest 0, 45 or 90 degree line.
VECTOR2< ret_type > GetClampedCoords(const VECTOR2< in_type > &aCoords, pad_type aPadding=1u)
Clamps a vector to values that can be negated, respecting numeric limits of coordinates data type wit...
void InferBold(TEXT_ATTRIBUTES *aAttrs)
Definition: gr_text.h:82
@ GRID_TEXT
Definition: grid_helper.h:49
@ GRID_GRAPHICS
Definition: grid_helper.h:50
bool IsBackLayer(PCB_LAYER_ID aLayerId)
Layer classification: check if it's a back layer.
Definition: layer_ids.h:788
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition: layer_ids.h:663
@ LAYER_ZONES
Control for copper zone opacity/visibility (color ignored).
Definition: layer_ids.h:294
@ LAYER_SHAPES
Copper graphic shape opacity/visibility (color ignored).
Definition: layer_ids.h:312
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ B_Cu
Definition: layer_ids.h:65
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
@ F_Cu
Definition: layer_ids.h:64
This file contains miscellaneous commonly used macros and functions.
#define KI_FALLTHROUGH
The KI_FALLTHROUGH macro is to be used when switch statement cases should purposely fallthrough from ...
Definition: macros.h:83
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
ZONE_MODE
Definition: pcb_actions.h:35
Class to handle a set of BOARD_ITEMs.
@ ID_POPUP_PCB_SELECT_CUSTOM_WIDTH
Definition: pcbnew_id.h:25
@ ID_POPUP_PCB_SELECT_VIASIZE1
Definition: pcbnew_id.h:44
@ ID_POPUP_PCB_SELECT_VIASIZE16
Definition: pcbnew_id.h:59
int GetUserUnits()
Return the currently selected user unit value for the interface.
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
Class that computes missing connections on a PCB.
@ RPT_SEVERITY_IGNORE
std::optional< VECTOR2I > OPT_VECTOR2I
Definition: seg.h:39
bool NoPrintableChars(const wxString &aString)
Return true if the string is empty or contains only whitespace.
LINE_STYLE
Dashed line types.
Definition: stroke_params.h:46
constexpr int mmToIU(double mm) const
Definition: base_units.h:88
MAGNETIC_OPTIONS tracks
MAGNETIC_OPTIONS pads
A filename or source description, a problem input line, a line number, a byte offset,...
Definition: ki_exception.h:120
Container to handle a stock of specific vias each with unique diameter and drill sizes in the BOARD c...
Parameters used to fully describe a zone creation process.
ZONE_MODE m_mode
Zone settings source (for similar and cutout zones)
bool m_keepout
< Should create a keepout zone?
ZONE * m_sourceZone
Zone leader mode.
PCB_LAYER_ID m_layer
The zone mode to operate in.
SHAPE_CIRCLE circle(c.m_circle_center, c.m_circle_radius)
int clearance
constexpr int delta
@ GR_TEXT_H_ALIGN_LEFT
@ GR_TEXT_V_ALIGN_BOTTOM
@ GR_TEXT_V_ALIGN_TOP
@ CMENU_NOW
Right now (after TOOL_INTERACTIVE::SetContextMenu).
Definition: tool_event.h:153
std::optional< TOOL_EVENT > OPT_TOOL_EVENT
Definition: tool_event.h:633
@ TA_CHOICE_MENU_CHOICE
Context menu choice.
Definition: tool_event.h:98
@ TA_CHOICE_MENU_UPDATE
Context menu update.
Definition: tool_event.h:94
@ TA_CHOICE_MENU_CLOSED
Context menu is closed, no matter whether anything has been chosen or not.
Definition: tool_event.h:101
@ MD_CTRL
Definition: tool_event.h:144
@ MD_SHIFT
Definition: tool_event.h:143
@ BUT_LEFT
Definition: tool_event.h:132
@ BUT_RIGHT
Definition: tool_event.h:133
bool TestSegmentHit(const VECTOR2I &aRefPoint, const VECTOR2I &aStart, const VECTOR2I &aEnd, int aDist)
Test if aRefPoint is with aDistance on the line defined by aStart and aEnd.
Definition: trigo.cpp:175
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
Definition: typeinfo.h:78
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
Definition: typeinfo.h:88
@ PCB_DIM_ORTHOGONAL_T
class PCB_DIM_ORTHOGONAL, a linear dimension constrained to x/y
Definition: typeinfo.h:105
@ PCB_DIM_LEADER_T
class PCB_DIM_LEADER, a leader dimension (graphic item)
Definition: typeinfo.h:102
@ PCB_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_ZONE_T
class ZONE, a copper pour area
Definition: typeinfo.h:107
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
Definition: typeinfo.h:92
@ PCB_REFERENCE_IMAGE_T
class PCB_REFERENCE_IMAGE, bitmap on a layer
Definition: typeinfo.h:89
@ PCB_FIELD_T
class PCB_FIELD, text associated with a footprint property
Definition: typeinfo.h:90
@ 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_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_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
VECTOR2< double > VECTOR2D
Definition: vector2d.h:694