KiCad PCB EDA Suite
Loading...
Searching...
No Matches
symbol_editor_move_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) 2019 CERN
5 * Copyright (C) 2019-2023 KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
25#include <tool/tool_manager.h>
27#include <ee_actions.h>
28#include <ee_grid_helper.h>
29#include <eda_item.h>
31#include <sch_shape.h>
32#include <sch_commit.h>
33#include <wx/log.h>
34#include <view/view_controls.h>
37
38
40 EE_TOOL_BASE( "eeschema.SymbolMoveTool" ),
41 m_moveInProgress( false )
42{
43}
44
45
47{
49
50 //
51 // Add move actions to the selection tool menu
52 //
54
55 auto canMove =
56 [&]( const SELECTION& sel )
57 {
59 wxCHECK( editor, false );
60
61 if( !editor->IsSymbolEditable() )
62 return false;
63
64 if( editor->IsSymbolAlias() )
65 {
66 for( EDA_ITEM* item : sel )
67 {
68 if( item->Type() != SCH_FIELD_T )
69 return false;
70 }
71 }
72
73 return true;
74 };
75
76 selToolMenu.AddItem( EE_ACTIONS::move, canMove && EE_CONDITIONS::IdleSelection, 150 );
77 selToolMenu.AddItem( EE_ACTIONS::alignToGrid, canMove && EE_CONDITIONS::IdleSelection, 150 );
78
79 return true;
80}
81
82
84{
85 EE_TOOL_BASE::Reset( aReason );
86
87 if( aReason == MODEL_RELOAD )
88 m_moveInProgress = false;
89}
90
91
93{
94 if( SCH_COMMIT* commit = dynamic_cast<SCH_COMMIT*>( aEvent.Commit() ) )
95 {
96 wxCHECK( aEvent.SynchronousState(), 0 );
97 aEvent.SynchronousState()->store( STS_RUNNING );
98
99 if( doMoveSelection( aEvent, commit ) )
100 aEvent.SynchronousState()->store( STS_FINISHED );
101 else
102 aEvent.SynchronousState()->store( STS_CANCELLED );
103 }
104 else
105 {
106 SCH_COMMIT localCommit( m_toolMgr );
107
108 if( doMoveSelection( aEvent, &localCommit ) )
109 localCommit.Push( _( "Move" ) );
110 else
111 localCommit.Revert();
112 }
113
114 return 0;
115}
116
117
119{
122
123 m_anchorPos = { 0, 0 };
124
125 // Be sure that there is at least one item that we can move. If there's no selection try
126 // looking for the stuff under mouse cursor (i.e. Kicad old-style hover selection).
127 EE_SELECTION& selection = m_frame->IsSymbolAlias()
130 bool unselect = selection.IsHover();
131
132 if( !m_frame->IsSymbolEditable() || selection.Empty() )
133 return false;
134
135 if( m_moveInProgress )
136 {
137 // The tool hotkey is interpreted as a click when already moving
139 return true;
140 }
141
142 m_frame->PushTool( aEvent );
143
144 Activate();
145 // Must be done after Activate() so that it gets set into the correct context
146 controls->ShowCursor( true );
147 controls->SetAutoPan( true );
148
149 bool restore_state = false;
150 TOOL_EVENT copy = aEvent;
151 TOOL_EVENT* evt = &copy;
152 VECTOR2I prevPos;
153 VECTOR2I moveOffset;
154
155 if( !selection.Front()->IsNew() )
156 aCommit->Modify( m_frame->GetCurSymbol(), m_frame->GetScreen() );
157
158 m_cursor = controls->GetCursorPosition( !aEvent.DisableGridSnapping() );
159
160 // Main loop: keep receiving events
161 do
162 {
163 m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::MOVING );
164 grid.SetSnap( !evt->Modifier( MD_SHIFT ) );
165 grid.SetUseGrid( getView()->GetGAL()->GetGridSnapping() && !evt->DisableGridSnapping() );
166
167 if( evt->IsAction( &EE_ACTIONS::move )
168 || evt->IsMotion()
169 || evt->IsDrag( BUT_LEFT )
171 {
172 if( !m_moveInProgress ) // Prepare to start moving/dragging
173 {
174 SCH_ITEM* lib_item = static_cast<SCH_ITEM*>( selection.Front() );
175
176 // Pick up any synchronized pins
177 //
178 // Careful when pasting. The pasted pin will be at the same location as it
179 // was copied from, leading us to believe it's a synchronized pin. It's not.
180 if( m_frame->SynchronizePins() && !( lib_item->GetEditFlags() & IS_PASTED ) )
181 {
182 std::set<SCH_PIN*> sync_pins;
183
184 for( EDA_ITEM* sel_item : selection )
185 {
186 lib_item = static_cast<SCH_ITEM*>( sel_item );
187
188 if( lib_item->Type() == SCH_PIN_T )
189 {
190 SCH_PIN* cur_pin = static_cast<SCH_PIN*>( lib_item );
191 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
192 std::vector<bool> got_unit( symbol->GetUnitCount() + 1 );
193
194 got_unit[cur_pin->GetUnit()] = true;
195
196 for( SCH_PIN* pin : symbol->GetAllLibPins() )
197 {
198 if( !got_unit[pin->GetUnit()]
199 && pin->GetPosition() == cur_pin->GetPosition()
200 && pin->GetOrientation() == cur_pin->GetOrientation()
201 && pin->GetBodyStyle() == cur_pin->GetBodyStyle()
202 && pin->GetType() == cur_pin->GetType()
203 && pin->GetName() == cur_pin->GetName() )
204 {
205 if( sync_pins.insert( pin ).second )
206 got_unit[pin->GetUnit()] = true;
207 }
208 }
209 }
210 }
211
212 for( SCH_PIN* pin : sync_pins )
213 m_selectionTool->AddItemToSel( pin, true /*quiet mode*/ );
214 }
215
216 // Apply any initial offset in case we're coming from a previous command.
217 //
218 for( EDA_ITEM* item : selection )
219 moveItem( item, moveOffset );
220
221 // Set up the starting position and move/drag offset
222 //
223 m_cursor = controls->GetCursorPosition( !evt->DisableGridSnapping() );
224
225 if( lib_item->IsNew() )
226 {
227 m_anchorPos = selection.GetReferencePoint();
229
230 // Drag items to the current cursor position
231 for( EDA_ITEM* item : selection )
232 {
233 moveItem( item, delta );
234 updateItem( item, false );
235 }
236
238 }
239 else if( m_frame->GetMoveWarpsCursor() )
240 {
241 VECTOR2I itemPos = selection.GetTopLeftItem()->GetPosition();
242 m_anchorPos = VECTOR2I( itemPos.x, itemPos.y );
243
246 }
247 else
248 {
249 m_cursor = controls->GetCursorPosition( !evt->DisableGridSnapping() );
251 }
252
253 controls->SetCursorPosition( m_cursor, false );
254
255 prevPos = m_cursor;
256 controls->SetAutoPan( true );
257 m_moveInProgress = true;
258 }
259
260 //------------------------------------------------------------------------
261 // Follow the mouse
262 //
263 m_cursor = grid.BestSnapAnchor( controls->GetCursorPosition( false ),
264 grid.GetSelectionGrid( selection ), selection );
265 VECTOR2I delta( m_cursor - prevPos );
267
268 moveOffset += delta;
269 prevPos = m_cursor;
270
271 for( EDA_ITEM* item : selection )
272 {
273 moveItem( item, delta );
274 updateItem( item, false );
275 }
276
278 }
279 //------------------------------------------------------------------------
280 // Handle cancel
281 //
282 else if( evt->IsCancelInteractive() || evt->IsActivate() )
283 {
284 if( m_moveInProgress )
285 {
286 evt->SetPassEvent( false );
287 restore_state = true;
288 }
289
290 break;
291 }
292 //------------------------------------------------------------------------
293 // Handle TOOL_ACTION special cases
294 //
295 else if( evt->Action() == TA_UNDO_REDO_PRE )
296 {
297 unselect = true;
298 break;
299 }
300 else if( evt->IsAction( &ACTIONS::doDelete ) )
301 {
302 // Exit on a remove operation; there is no further processing for removed items.
303 break;
304 }
305 else if( evt->IsAction( &ACTIONS::duplicate ) )
306 {
307 wxBell();
308 }
309 //------------------------------------------------------------------------
310 // Handle context menu
311 //
312 else if( evt->IsClick( BUT_RIGHT ) )
313 {
315 }
316 //------------------------------------------------------------------------
317 // Handle drop
318 //
319 else if( evt->IsMouseUp( BUT_LEFT )
320 || evt->IsClick( BUT_LEFT )
321 || evt->IsDblClick( BUT_LEFT ) )
322 {
323 if( selection.GetSize() == 1 && selection.Front()->Type() == SCH_PIN_T )
324 {
326
327 try
328 {
329 SCH_PIN* curr_pin = static_cast<SCH_PIN*>( selection.Front() );
330
331 if( pinTool->PlacePin( curr_pin ) )
332 {
333 // PlacePin() clears the current selection, which we don't want. Not only
334 // is it a poor user experience, but it also prevents us from doing the
335 // proper cleanup at the end of this routine (ie: clearing the edit flags).
336 m_selectionTool->AddItemToSel( curr_pin, true /*quiet mode*/ );
337 }
338 else
339 {
340 restore_state = true;
341 }
342 }
343 catch( const boost::bad_pointer& e )
344 {
345 restore_state = true;
346 wxLogError( "Boost pointer exception occurred: \"%s\"", e.what() );
347 }
348 }
349
350 break; // Finish
351 }
352 else
353 {
354 evt->SetPassEvent();
355 }
356
357 } while( ( evt = Wait() ) ); // Assignment intentional; not equality test
358
359 controls->ForceCursorPosition( false );
360 controls->ShowCursor( false );
361 controls->SetAutoPan( false );
362
363 m_anchorPos = { 0, 0 };
364
365 for( EDA_ITEM* item : selection )
366 item->ClearEditFlags();
367
368 if( unselect )
370
371 m_moveInProgress = false;
372 m_frame->PopTool( aEvent );
373
374 return !restore_state;
375}
376
377
379{
382 SCH_COMMIT commit( m_toolMgr );
383
384 auto doMoveItem =
385 [&]( EDA_ITEM* item, const VECTOR2I& delta )
386 {
387 commit.Modify( item, m_frame->GetScreen() );
388 static_cast<SCH_ITEM*>( item )->Move( delta );
389 updateItem( item, true );
390 };
391
392 for( EDA_ITEM* item : selection )
393 {
394 if( SCH_SHAPE* shape = dynamic_cast<SCH_SHAPE*>( item ) )
395 {
396 VECTOR2I newStart = grid.AlignGrid( shape->GetStart(), grid.GetItemGrid( shape ) );
397 VECTOR2I newEnd = grid.AlignGrid( shape->GetEnd(), grid.GetItemGrid( shape ) );
398
399 switch( shape->GetShape() )
400 {
401 case SHAPE_T::SEGMENT:
402 case SHAPE_T::RECTANGLE:
403 case SHAPE_T::CIRCLE:
404 case SHAPE_T::ARC:
405 if( newStart == newEnd )
406 {
407 // Don't collapse shape; just snap its position
408 if( newStart != shape->GetStart() )
409 doMoveItem( shape, newStart - shape->GetStart() );
410 }
411 else if( newStart != shape->GetStart() || newEnd != shape->GetEnd() )
412 {
413 // Snap both ends
414 commit.Modify( shape, m_frame->GetScreen() );
415
416 shape->SetStart( newStart );
417 shape->SetEnd( newEnd );
418
419 updateItem( item, true );
420 }
421
422 break;
423
424 case SHAPE_T::POLY:
425 if( shape->GetPointCount() > 0 )
426 {
427 std::vector<VECTOR2I> newPts;
428
429 for( const VECTOR2I& pt : shape->GetPolyShape().Outline( 0 ).CPoints() )
430 newPts.push_back( grid.AlignGrid( pt, grid.GetItemGrid( shape ) ) );
431
432 bool collapsed = false;
433
434 for( int ii = 0; ii < (int) newPts.size() - 1; ++ii )
435 {
436 if( newPts[ii] == newPts[ii + 1] )
437 collapsed = true;
438 }
439
440 if( collapsed )
441 {
442 // Don't collapse shape; just snap its position
443 if( newStart != shape->GetStart() )
444 doMoveItem( shape, newStart - shape->GetStart() );
445 }
446 else
447 {
448 commit.Modify( shape, m_frame->GetScreen() );
449
450 for( int ii = 0; ii < (int) newPts.size(); ++ii )
451 shape->GetPolyShape().Outline( 0 ).SetPoint( ii, newPts[ii] );
452
453 updateItem( item, true );
454 }
455 }
456
457 break;
458
459 case SHAPE_T::BEZIER:
460 // Snapping bezier control points is unlikely to be useful. Just snap its
461 // position.
462 if( newStart != shape->GetStart() )
463 doMoveItem( shape, newStart - shape->GetStart() );
464
465 break;
466
467 case SHAPE_T::UNDEFINED:
468 wxASSERT_MSG( false, wxT( "Undefined shape in AlignElements" ) );
469 break;
470 }
471 }
472 else
473 {
474 VECTOR2I newPos = grid.AlignGrid( item->GetPosition(), grid.GetItemGrid( item ) );
475 VECTOR2I delta = newPos - item->GetPosition();
476
477 if( delta != VECTOR2I( 0, 0 ) )
478 doMoveItem( item, delta );
479
480 if( SCH_PIN* pin = dynamic_cast<SCH_PIN*>( item ) )
481 {
482 int length = pin->GetLength();
483 int pinGrid;
484
485 if( pin->GetOrientation() == PIN_ORIENTATION::PIN_LEFT
486 || pin->GetOrientation() == PIN_ORIENTATION::PIN_RIGHT )
487 {
488 pinGrid = KiROUND( grid.GetGridSize( grid.GetItemGrid( item ) ).x );
489 }
490 else
491 {
492 pinGrid = KiROUND( grid.GetGridSize( grid.GetItemGrid( item ) ).y );
493 }
494
495 int newLength = KiROUND( (double) length / pinGrid ) * pinGrid;
496
497 if( newLength > 0 )
498 pin->SetLength( newLength );
499 }
500 }
501 }
502
504
505 commit.Push( _( "Align" ) );
506 return 0;
507}
508
509
511{
512 static_cast<SCH_ITEM*>( aItem )->Move( aDelta );
513 aItem->SetFlags( IS_MOVING );
514}
515
516
518{
521}
static TOOL_ACTION duplicate
Definition: actions.h:75
static TOOL_ACTION doDelete
Definition: actions.h:76
static TOOL_ACTION cursorClick
Definition: actions.h:158
static TOOL_ACTION refreshPreview
Definition: actions.h:138
COMMIT & Modify(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Create an undo entry for an item that has been already modified.
Definition: commit.h:105
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 SetCurrentCursor(KICURSOR aCursor)
Set the current cursor shape for this panel.
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:88
virtual VECTOR2I GetPosition() const
Definition: eda_item.h:242
EDA_ITEM_FLAGS GetEditFlags() const
Definition: eda_item.h:132
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition: eda_item.h:126
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:100
bool IsNew() const
Definition: eda_item.h:106
static TOOL_ACTION alignToGrid
Definition: ee_actions.h:120
static TOOL_ACTION move
Definition: ee_actions.h:121
static TOOL_ACTION clearSelection
Clears the current selection.
Definition: ee_actions.h:56
EE_SELECTION & RequestSelection(const std::vector< KICAD_T > &aScanTypes={ SCH_LOCATE_ANY_T }, bool aPromoteCellSelections=false)
Return either an existing selection (filtered), or the selection at the current cursor position if th...
EE_SELECTION & GetSelection()
A foundation class for a tool operating on a schematic or symbol.
Definition: ee_tool_base.h:48
void Reset(RESET_REASON aReason) override
Bring the tool to a known, initial state.
Definition: ee_tool_base.h:84
void updateItem(EDA_ITEM *aItem, bool aUpdateRTree) const
Similar to getView()->Update(), but handles items that are redrawn by their parents and updating the ...
Definition: ee_tool_base.h:109
EE_SELECTION_TOOL * m_selectionTool
Definition: ee_tool_base.h:200
bool Init() override
Init() is called once upon a registration of the tool.
Definition: ee_tool_base.h:64
static const TOOL_EVENT SelectedItemsMoved
Used to inform tools that the selection should temporarily be non-editable.
Definition: actions.h:274
An interface for classes handling user events controlling the view behavior such as zooming,...
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 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.
Define a library symbol object.
Definition: lib_symbol.h:77
std::vector< SCH_PIN * > GetAllLibPins() const
Return a list of pin pointers for all units / converts.
Definition: lib_symbol.cpp:840
int GetUnitCount() const override
SCH_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
virtual void Push(const wxString &aMessage=wxT("A commit"), int aCommitFlags=0) override
Revert the commit by restoring the modified items state.
Definition: sch_commit.cpp:406
virtual void Revert() override
Definition: sch_commit.cpp:484
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:166
int GetBodyStyle() const
Definition: sch_item.h:232
int GetUnit() const
Definition: sch_item.h:229
const wxString & GetName() const
Definition: sch_pin.cpp:353
PIN_ORIENTATION GetOrientation() const
Definition: sch_pin.cpp:245
VECTOR2I GetPosition() const override
Definition: sch_pin.cpp:237
ELECTRICAL_PINTYPE GetType() const
Definition: sch_pin.cpp:291
static bool IdleSelection(const SELECTION &aSelection)
Test if all selected items are not being edited.
int AddItemToSel(const TOOL_EVENT &aEvent)
bool IsHover() const
Definition: selection.h:84
virtual unsigned int GetSize() const override
Return the number of stored items.
Definition: selection.h:100
EDA_ITEM * Front() const
Definition: selection.h:172
bool Empty() const
Checks if there is anything selected.
Definition: selection.h:110
bool doMoveSelection(const TOOL_EVENT &aEvent, SCH_COMMIT *aCommit)
void Reset(RESET_REASON aReason) override
Bring the tool to a known, initial state.
void setTransitions() override
This method is meant to be overridden in order to specify handlers for events.
void moveItem(EDA_ITEM *aItem, const VECTOR2I &aDelta)
Set up handlers for various events.
bool Init() override
Init() is called once upon a registration of the tool.
int Main(const TOOL_EVENT &aEvent)
Run an interactive move of the selected items, or the item under the cursor.
int AlignElements(const TOOL_EVENT &aEvent)
Align selected elements to the grid.
bool m_moveInProgress
Last cursor position (needed for getModificationPoint() to avoid changes of edit reference point).
The symbol library editor main window.
bool IsSymbolAlias() const
Return true if aLibId is an alias for the editor screen symbol.
bool IsSymbolEditable() const
Test if a symbol is loaded and can be edited.
LIB_SYMBOL * GetCurSymbol() const
Return the current symbol being edited or NULL if none selected.
virtual void PopTool(const TOOL_EVENT &aEvent)
Pops a tool from the stack.
bool GetMoveWarpsCursor() const
Indicate that a move operation should warp the mouse pointer to the origin of the move object.
Definition: tools_holder.h:150
virtual void PushTool(const TOOL_EVENT &aEvent)
NB: the definition of "tool" is different at the user level.
KIGFX::VIEW_CONTROLS * getViewControls() const
Return the instance of VIEW_CONTROLS object used in the application.
Definition: tool_base.cpp:42
TOOL_MANAGER * m_toolMgr
Definition: tool_base.h:217
KIGFX::VIEW * getView() const
Returns the instance of #VIEW object used in the application.
Definition: tool_base.cpp:36
RESET_REASON
Determine the reason of reset for a tool.
Definition: tool_base.h:78
@ MODEL_RELOAD
Model changes (the sheet for a schematic)
Definition: tool_base.h:80
Generic, UI-independent tool event.
Definition: tool_event.h:167
bool DisableGridSnapping() const
Definition: tool_event.h:363
bool IsCancelInteractive() const
Indicate the event should restart/end an ongoing interactive tool's event loop (eg esc key,...
Definition: tool_event.cpp:221
TOOL_ACTIONS Action() const
These give a tool a method of informing the TOOL_MANAGER that a particular event should be passed on ...
Definition: tool_event.h:246
bool IsActivate() const
Definition: tool_event.h:337
COMMIT * Commit() const
Returns information about difference between current mouse cursor position and the place where draggi...
Definition: tool_event.h:275
bool IsClick(int aButtonMask=BUT_ANY) const
Definition: tool_event.cpp:209
bool IsDrag(int aButtonMask=BUT_ANY) const
Definition: tool_event.h:307
int Modifier(int aMask=MD_MODIFIER_MASK) const
Definition: tool_event.h:358
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
bool IsDblClick(int aButtonMask=BUT_ANY) const
Definition: tool_event.cpp:215
std::atomic< SYNCRONOUS_TOOL_STATE > * SynchronousState() const
Definition: tool_event.h:272
void SetPassEvent(bool aPass=true)
Returns if it this event has a valid position (true for mouse events and context-menu or hotkey-based...
Definition: tool_event.h:252
bool IsMouseUp(int aButtonMask=BUT_ANY) const
Definition: tool_event.h:317
bool IsMotion() const
Definition: tool_event.h:322
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).
TOOL_MENU & GetToolMenu()
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 PostEvent(const TOOL_EVENT &aEvent)
Put an event to the event queue to be processed at the end of event processing cycle.
bool RunAction(const std::string &aActionName, T aParam)
Run the specified action immediately, pausing the current action to run the new one.
Definition: tool_manager.h:150
CONDITIONAL_MENU & GetMenu()
Definition: tool_menu.cpp:44
void ShowContextMenu(SELECTION &aSelection)
Helper function to set and immediately show a CONDITIONAL_MENU in concert with the given SELECTION.
Definition: tool_menu.cpp:57
#define _(s)
#define IS_PASTED
Modifier on IS_NEW which indicates it came from clipboard.
#define IS_MOVING
Item being moved.
constexpr int delta
@ TA_UNDO_REDO_PRE
Definition: tool_event.h:105
@ MD_SHIFT
Definition: tool_event.h:142
@ STS_CANCELLED
Definition: tool_event.h:160
@ STS_FINISHED
Definition: tool_event.h:159
@ STS_RUNNING
Definition: tool_event.h:158
@ BUT_LEFT
Definition: tool_event.h:131
@ BUT_RIGHT
Definition: tool_event.h:132
@ SCH_FIELD_T
Definition: typeinfo.h:150
@ SCH_PIN_T
Definition: typeinfo.h:153
constexpr ret_type KiROUND(fp_type v)
Round a floating point number to an integer using "round halfway cases away from zero".
Definition: util.h:121
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:673