KiCad PCB EDA Suite
Loading...
Searching...
No Matches
tool_manager.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2013-2023 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Tomasz Wlostowski <[email protected]>
7 * @author Maciej Suminski <[email protected]>
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, you may find one here:
21 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
22 * or you may search the http://www.gnu.org website for the version 2 license,
23 * or you may write to the Free Software Foundation, Inc.,
24 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
25 */
26
27#include <core/kicad_algo.h>
28#include <scoped_set_reset.h>
29#include <optional>
30#include <map>
31#include <stack>
32#include <trace_helpers.h>
33#include <kiplatform/ui.h>
34#include <app_monitor.h>
35
36#include <wx/event.h>
37#include <wx/evtloop.h>
38#include <wx/clipbrd.h>
39#include <wx/app.h>
40
41#include <math/vector2wx.h>
42
43#include <view/view.h>
44#include <view/view_controls.h>
45#include <eda_base_frame.h>
46#include <tool/tool_base.h>
48#include <tool/tool_manager.h>
49#include <tool/action_menu.h>
50#include <tool/coroutine.h>
51#include <tool/action_manager.h>
52
54
57{
59 theTool( aTool )
60 {
61 clear();
62 }
63
64 TOOL_STATE( const TOOL_STATE& aState )
65 {
66 theTool = aState.theTool;
67 idle = aState.idle;
68 shutdown = aState.shutdown;
69 pendingWait = aState.pendingWait;
71 contextMenu = aState.contextMenu;
73 cofunc = aState.cofunc;
75 wakeupEvent = aState.wakeupEvent;
76 waitEvents = aState.waitEvents;
77 transitions = aState.transitions;
78 vcSettings = aState.vcSettings;
79 // do not copy stateStack
80 }
81
83 {
84 wxASSERT_MSG( stateStack.empty(), wxT( "StateStack not empty!" ) );
85 }
86
89
91 bool idle;
92
95
99
102
105
108
111
114
117
120
123 std::vector<TRANSITION> transitions;
124
127
129 {
130 theTool = aState.theTool;
131 idle = aState.idle;
132 shutdown = aState.shutdown;
133 pendingWait = aState.pendingWait;
135 contextMenu = aState.contextMenu;
137 cofunc = aState.cofunc;
138 initialEvent = aState.initialEvent;
139 wakeupEvent = aState.wakeupEvent;
140 waitEvents = aState.waitEvents;
141 transitions = aState.transitions;
142 vcSettings = aState.vcSettings;
143
144 // do not copy stateStack
145 return *this;
146 }
147
148 bool operator==( const TOOL_MANAGER::TOOL_STATE& aRhs ) const
149 {
150 return aRhs.theTool == theTool;
151 }
152
153 bool operator!=( const TOOL_MANAGER::TOOL_STATE& aRhs ) const
154 {
155 return aRhs.theTool != theTool;
156 }
157
162 void Push()
163 {
164 auto state = std::make_unique<TOOL_STATE>( *this );
165 stateStack.push( std::move( state ) );
166 clear();
167 }
168
175 bool Pop()
176 {
177 delete cofunc;
178
179 if( !stateStack.empty() )
180 {
181 *this = *stateStack.top().get();
182 stateStack.pop();
183 return true;
184 }
185
187 return false;
188 }
189
190private:
192 std::stack<std::unique_ptr<TOOL_STATE>> stateStack;
193
196 {
197 cofunc = nullptr;
198 shutdown = false;
199 pendingWait = false;
200 pendingContextMenu = false;
201 contextMenu = nullptr;
203 }
204
206 void clear()
207 {
208 idle = true;
210 vcSettings.Reset();
211 transitions.clear();
212 }
213};
214
215
217 m_model( nullptr ),
218 m_view( nullptr ),
219 m_viewControls( nullptr ),
220 m_frame( nullptr ),
221 m_settings( nullptr ),
223 m_menuActive( false ),
224 m_menuOwner( -1 ),
225 m_activeState( nullptr ),
226 m_shuttingDown( false )
227{
228 m_actionMgr = new ACTION_MANAGER( this );
229}
230
231
233{
234 std::map<TOOL_BASE*, TOOL_STATE*>::iterator it, it_end;
235
236 for( it = m_toolState.begin(), it_end = m_toolState.end(); it != it_end; ++it )
237 {
238 delete it->second->cofunc; // delete cofunction
239 delete it->second; // delete TOOL_STATE
240 delete it->first; // delete the tool itself
241 }
242
243 delete m_actionMgr;
244}
245
246
248{
249 wxASSERT_MSG( m_toolNameIndex.find( aTool->GetName() ) == m_toolNameIndex.end(),
250 wxT( "Adding two tools with the same name may result in unexpected behavior.") );
251 wxASSERT_MSG( m_toolIdIndex.find( aTool->GetId() ) == m_toolIdIndex.end(),
252 wxT( "Adding two tools with the same ID may result in unexpected behavior.") );
253 wxASSERT_MSG( m_toolTypes.find( typeid( *aTool ).name() ) == m_toolTypes.end(),
254 wxT( "Adding two tools of the same type may result in unexpected behavior.") );
255
256 wxLogTrace( kicadTraceToolStack,
257 wxS( "TOOL_MANAGER::RegisterTool: Registering tool %s with ID %d" ),
258 aTool->GetName(), aTool->GetId() );
259
260 m_toolOrder.push_back( aTool );
261
262 TOOL_STATE* st = new TOOL_STATE( aTool );
263
264 m_toolState[aTool] = st;
265 m_toolNameIndex[aTool->GetName()] = st;
266 m_toolIdIndex[aTool->GetId()] = st;
267 m_toolTypes[typeid( *aTool ).name()] = st->theTool;
268
269 aTool->attachManager( this );
270}
271
272
274{
275 TOOL_BASE* tool = FindTool( aToolId );
276
277 if( tool && tool->GetType() == INTERACTIVE )
278 return invokeTool( tool );
279
280 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::InvokeTool - no tool with ID %d" ),
281 aToolId );
282
283 return false; // there is no tool with the given id
284}
285
286
287bool TOOL_MANAGER::InvokeTool( const std::string& aToolName )
288{
289 TOOL_BASE* tool = FindTool( aToolName );
290
291 if( tool && tool->GetType() == INTERACTIVE )
292 return invokeTool( tool );
293
294 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::InvokeTool - no tool with name %s" ),
295 aToolName );
296
297 return false; // there is no tool with the given name
298}
299
300
301bool TOOL_MANAGER::doRunAction( const std::string& aActionName, bool aNow, const ki::any& aParam,
302 COMMIT* aCommit )
303{
304 TOOL_ACTION* action = m_actionMgr->FindAction( aActionName );
305
306 if( !action )
307 {
308 wxASSERT_MSG( false, wxString::Format( "Could not find action %s.", aActionName ) );
309 return false;
310 }
311
312 doRunAction( *action, aNow, aParam, aCommit );
313
314 return true;
315}
316
317
319{
320 if( m_viewControls )
321 return m_viewControls->GetMousePosition();
322 else
324}
325
326
328{
329 if( m_viewControls )
330 return m_viewControls->GetCursorPosition();
331 else
333}
334
335
336bool TOOL_MANAGER::doRunAction( const TOOL_ACTION& aAction, bool aNow, const ki::any& aParam,
337 COMMIT* aCommit, bool aFromAPI )
338{
339 if( m_shuttingDown )
340 return true;
341
342 bool retVal = false;
343 TOOL_EVENT event = aAction.MakeEvent();
344
345 if( event.Category() == TC_COMMAND )
346 event.SetMousePosition( m_hotKeyPos.value_or( GetCursorPosition() ) );
347
348 // Allow to override the action parameter
349 if( aParam.has_value() )
350 event.SetParameter( aParam );
351
352 if( aNow )
353 {
354 TOOL_STATE* current = m_activeState;
355
356 // An event with a commit must be run synchronously
357 if( aCommit )
358 {
359 // We initialize the SYNCHRONOUS state to finished so that tools that don't have an
360 // event loop won't hang if someone forgets to set the state.
361 std::atomic<SYNCRONOUS_TOOL_STATE> synchronousControl = STS_FINISHED;
362
363 event.SetSynchronous( &synchronousControl );
364 event.SetCommit( aCommit );
365
366 processEvent( event );
367
368 while( synchronousControl == STS_RUNNING )
369 {
370 wxYield(); // Needed to honor mouse (and other) events during editing
371 wxMilliSleep( 1 ); // Needed to avoid 100% use of one cpu core.
372 // The sleeping time must be must be small to avoid
373 // noticeable lag in mouse and editing events
374 // (1 to 5 ms is a good value)
375 }
376
377 retVal = synchronousControl != STS_CANCELLED;
378 }
379 else
380 {
381 retVal = processEvent( event );
382 }
383
384 setActiveState( current );
385 UpdateUI( event );
386 }
387 else
388 {
389 // It is really dangerous to pass a commit (whose lifetime we can't guarantee) to
390 // deferred event processing. There is a possibility that user actions will get run
391 // in between, which might either affect the lifetime of the commit or push or pop
392 // other commits. However, we don't currently have a better solution for the API.
393 if( aCommit )
394 {
395 wxASSERT_MSG( aFromAPI, wxT( "Deferred actions have no way of guaranteeing the "
396 "lifetime of the COMMIT object" ) );
397 event.SetCommit( aCommit );
398 }
399
400 PostEvent( event );
401 }
402
403 return retVal;
404}
405
406
408{
410
411 processEvent( evt );
412}
413
414
415void TOOL_MANAGER::PrimeTool( const VECTOR2D& aPosition )
416{
417 int modifiers = 0;
418
419 /*
420 * Don't include any modifiers. They're part of the hotkey, not part of the resulting
421 * click.
422 *
423 * modifiers |= wxGetKeyState( WXK_SHIFT ) ? MD_SHIFT : 0;
424 * modifiers |= wxGetKeyState( WXK_CONTROL ) ? MD_CTRL : 0;
425 * modifiers |= wxGetKeyState( WXK_ALT ) ? MD_ALT : 0;
426 */
427
428 TOOL_EVENT evt( TC_MOUSE, TA_PRIME, BUT_LEFT | modifiers );
429 evt.SetMousePosition( aPosition );
430
431 PostEvent( evt );
432}
433
434
436{
437 // Horrific hack, but it's a crash bug. Don't let inter-frame commands stack up
438 // waiting to be processed.
439 if( aEvent.IsSimulator() && m_eventQueue.size() > 0 && m_eventQueue.back().IsSimulator() )
440 m_eventQueue.pop_back();
441
442 m_eventQueue.push_back( aEvent );
443}
444
445
446int TOOL_MANAGER::GetHotKey( const TOOL_ACTION& aAction ) const
447{
448 return m_actionMgr->GetHotKey( aAction );
449}
450
451
453{
454 wxASSERT( aTool != nullptr );
455
456 TOOL_EVENT evt( TC_COMMAND, TA_ACTIVATE, aTool->GetName() );
458 processEvent( evt );
459
460 if( TOOL_STATE* active = GetCurrentToolState() )
461 setActiveState( active );
462
463 return true;
464}
465
466
468{
469 wxASSERT( aTool != nullptr );
470
471 wxString msg = wxString::Format( wxS( "TOOL_MANAGER::runTool - running tool %s" ), aTool->GetName() );
472 APP_MONITOR::AddTransactionBreadcrumb( msg, "tool.run" );
473
474 if( !isRegistered( aTool ) )
475 {
476 wxASSERT_MSG( false, wxT( "You cannot run unregistered tools" ) );
477 return false;
478 }
479
480 TOOL_ID id = aTool->GetId();
481
482 wxLogTrace( kicadTraceToolStack, msg,
483 aTool->GetName() );
484
485 if( aTool->GetType() == INTERACTIVE )
486 static_cast<TOOL_INTERACTIVE*>( aTool )->resetTransitions();
487
488 // If the tool is already active, bring it to the top of the active tools stack
489 if( isActive( aTool ) && m_activeTools.size() > 1 )
490 {
491 auto it = std::find( m_activeTools.begin(), m_activeTools.end(), id );
492
493 if( it != m_activeTools.end() )
494 {
495 if( it != m_activeTools.begin() )
496 {
497 m_activeTools.erase( it );
498 m_activeTools.push_front( id );
499 }
500
501 return false;
502 }
503 }
504
507
508 // Add the tool on the front of the processing queue (it gets events first)
509 m_activeTools.push_front( id );
510
511 return true;
512}
513
514
516{
517 m_shuttingDown = true;
518
519 // Create a temporary list of tools to iterate over since when the tools shutdown
520 // they remove themselves from the list automatically (invalidating the iterator)
521 ID_LIST tmpList = m_activeTools;
522
523 // Make sure each tool knows that it is shutting down, so that loops get shut down
524 // at the dispatcher
525 for( auto id : tmpList )
526 {
527 if( m_toolIdIndex.count( id ) == 0 )
528 continue;
529
530 m_toolIdIndex[id]->shutdown = true;
531 }
532
533 for( auto id : tmpList )
534 {
535 ShutdownTool( id );
536 }
537}
538
539
541{
542 TOOL_BASE* tool = FindTool( aToolId );
543
544 if( tool && tool->GetType() == INTERACTIVE )
545 ShutdownTool( tool );
546
547 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::ShutdownTool - no tool with ID %d" ),
548 aToolId );
549}
550
551
552void TOOL_MANAGER::ShutdownTool( const std::string& aToolName )
553{
554 TOOL_BASE* tool = FindTool( aToolName );
555
556 if( tool && tool->GetType() == INTERACTIVE )
557 ShutdownTool( tool );
558
559 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::ShutdownTool - no tool with name %s" ),
560 aToolName );
561}
562
563
565{
566 wxASSERT( aTool != nullptr );
567
568 TOOL_ID id = aTool->GetId();
569
570 if( isActive( aTool ) )
571 {
572 TOOL_MANAGER::ID_LIST::iterator it = std::find( m_activeTools.begin(),
573 m_activeTools.end(), id );
574
575 TOOL_STATE* st = m_toolIdIndex[*it];
576
577 // the tool state handler is waiting for events (i.e. called Wait() method)
578 if( st && st->pendingWait )
579 {
580 // Wake up the tool and tell it to shutdown
581 st->shutdown = true;
582 st->pendingWait = false;
583 st->waitEvents.clear();
584
585 if( st->cofunc )
586 {
587 wxLogTrace( kicadTraceToolStack,
588 wxS( "TOOL_MANAGER::ShutdownTool - Shutting down tool %s" ),
589 st->theTool->GetName() );
590
591 setActiveState( st );
592 bool end = !st->cofunc->Resume();
593
594 if( end )
595 finishTool( st );
596 }
597 }
598 }
599}
600
601
603{
604 std::map<TOOL_ID, TOOL_STATE*>::const_iterator it = m_toolIdIndex.find( aId );
605
606 if( it != m_toolIdIndex.end() )
607 return it->second->theTool;
608
609 return nullptr;
610}
611
612
613TOOL_BASE* TOOL_MANAGER::FindTool( const std::string& aName ) const
614{
615 std::map<std::string, TOOL_STATE*>::const_iterator it = m_toolNameIndex.find( aName );
616
617 if( it != m_toolNameIndex.end() )
618 return it->second->theTool;
619
620 return nullptr;
621}
622
623
625{
626 // Deactivate the active tool, but do not run anything new
628 processEvent( evt );
629}
630
631
633{
634 if( aReason != TOOL_BASE::REDRAW )
636
637 for( auto& state : m_toolState )
638 {
639 TOOL_BASE* tool = state.first;
640
641 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::ResetTools: Resetting tool '%s'" ),
642 tool->GetName() );
643
644 setActiveState( state.second );
645 tool->Reset( aReason );
646
647 if( tool->GetType() == INTERACTIVE )
648 static_cast<TOOL_INTERACTIVE*>( tool )->resetTransitions();
649 }
650}
651
652
654{
655 for( auto it = m_toolOrder.begin(); it != m_toolOrder.end(); /* iter inside */ )
656 {
657 TOOL_BASE* tool = *it;
658 wxASSERT( m_toolState.count( tool ) );
659 TOOL_STATE* state = m_toolState[tool];
660 setActiveState( state );
661 ++it; // keep the iterator valid if the element is going to be erased
662
663 if( !tool->Init() )
664 {
665 wxLogTrace( kicadTraceToolStack,
666 wxS( "TOOL_MANAGER initialization of tool '%s' failed" ),
667 tool->GetName() );
668
669 // Unregister the tool
670 setActiveState( nullptr );
671 m_toolState.erase( tool );
672 m_toolNameIndex.erase( tool->GetName() );
673 m_toolIdIndex.erase( tool->GetId() );
674 m_toolTypes.erase( typeid( *tool ).name() );
675
676 delete state;
677 delete tool;
678 }
679 }
680
681 m_actionMgr->UpdateHotKeys( true );
682
684}
685
686
687int TOOL_MANAGER::GetPriority( int aToolId ) const
688{
689 int priority = 0;
690
691 for( TOOL_ID tool : m_activeTools )
692 {
693 if( tool == aToolId )
694 return priority;
695
696 ++priority;
697 }
698
699 return -1;
700}
701
702
704 const TOOL_EVENT_LIST& aConditions )
705{
706 TOOL_STATE* st = m_toolState[aTool];
707
708 st->transitions.emplace_back( TRANSITION( aConditions, aHandler ) );
709}
710
711
713{
714 m_toolState[aTool]->transitions.clear();
715}
716
717
718void TOOL_MANAGER::RunMainStack( TOOL_BASE* aTool, std::function<void()> aFunc )
719{
720 TOOL_STATE* st = m_toolState[aTool];
721 setActiveState( st );
722 wxCHECK( st->cofunc, /* void */ );
723 st->cofunc->RunMainStack( std::move( aFunc ) );
724}
725
726
728{
729 TOOL_STATE* st = m_toolState[aTool];
730
731 wxCHECK( !st->pendingWait, nullptr ); // everything collapses on two KiYield() in a row
732
733 // indicate to the manager that we are going to sleep and we shall be
734 // woken up when an event matching aConditions arrive
735 st->pendingWait = true;
736 st->waitEvents = aConditions;
737
738 wxCHECK( st->cofunc, nullptr );
739
740 // switch context back to event dispatcher loop
741 st->cofunc->KiYield();
742
743 // If the tool should shutdown, it gets a null event to break the loop
744 if( st->shutdown )
745 return nullptr;
746 else
747 return &st->wakeupEvent;
748}
749
750
752{
753 bool handled = false;
754
755 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::dispatchInternal - received event: %s" ),
756 aEvent.Format() );
757
758 auto it = m_activeTools.begin();
759
760 // iterate over active tool stack
761 while( it != m_activeTools.end() )
762 {
763 TOOL_STATE* st = m_toolIdIndex[*it];
764 bool increment = true;
765
766 // forward context menu events to the tool that created the menu
767 if( aEvent.IsChoiceMenu() )
768 {
769 if( *it != m_menuOwner )
770 {
771 ++it;
772 continue;
773 }
774 }
775
776 // If we're pendingWait then we had better have a cofunc to process the wait.
777 wxASSERT( !st || !st->pendingWait || st->cofunc );
778
779 // the tool state handler is waiting for events (i.e. called Wait() method)
780 if( st && st->cofunc && st->pendingWait && st->waitEvents.Matches( aEvent ) )
781 {
782 if( !aEvent.FirstResponder() )
783 aEvent.SetFirstResponder( st->theTool );
784
785 // got matching event? clear wait list and wake up the coroutine
786 st->wakeupEvent = aEvent;
787 st->pendingWait = false;
788 st->waitEvents.clear();
789
790 wxLogTrace( kicadTraceToolStack,
791 wxS( "TOOL_MANAGER::dispatchInternal - Waking tool %s for event: %s" ),
792 st->theTool->GetName(), aEvent.Format() );
793
794 setActiveState( st );
795 bool end = !st->cofunc->Resume();
796
797 if( end )
798 {
799 it = finishTool( st );
800 increment = false;
801 }
802
803 // If the tool did not request the event be passed to other tools, we're done
804 if( !st->wakeupEvent.PassEvent() )
805 {
806 wxLogTrace( kicadTraceToolStack,
807 wxS( "TOOL_MANAGER::dispatchInternal - tool %s stopped passing "
808 "event: %s" ),
809 st->theTool->GetName(), aEvent.Format() );
810
811 return true;
812 }
813 }
814
815 if( increment )
816 ++it;
817 }
818
819 for( const auto& state : m_toolState )
820 {
821 TOOL_STATE* st = state.second;
822 bool finished = false;
823
824 // no state handler in progress - check if there are any transitions (defined by
825 // Go() method that match the event.
826 if( !st->transitions.empty() )
827 {
828 for( const TRANSITION& tr : st->transitions )
829 {
830 if( tr.first.Matches( aEvent ) )
831 {
832 auto func_copy = tr.second;
833
834 if( !aEvent.FirstResponder() )
835 aEvent.SetFirstResponder( st->theTool );
836
837 // if there is already a context, then push it on the stack
838 // and transfer the previous view control settings to the new context
839 if( st->cofunc )
840 {
841 KIGFX::VC_SETTINGS viewControlSettings = st->vcSettings;
842 st->Push();
843 st->vcSettings = std::move( viewControlSettings );
844 }
845
846 st->cofunc = new COROUTINE<int, const TOOL_EVENT&>( std::move( func_copy ) );
847
848 wxLogTrace( kicadTraceToolStack,
849 wxS( "TOOL_MANAGER::dispatchInternal - Running tool %s for "
850 "event: %s" ),
851 st->theTool->GetName(), aEvent.Format() );
852
853 // got match? Run the handler.
854 setActiveState( st );
855 st->idle = false;
856 st->initialEvent = aEvent;
857 st->cofunc->Call( st->initialEvent );
858 handled = true;
859
860 if( !st->cofunc->Running() )
861 finishTool( st ); // The coroutine has finished immediately?
862
863 // if it is a message, continue processing
864 finished = !( aEvent.Category() == TC_MESSAGE );
865
866 // there is no point in further checking, as transitions got cleared
867 break;
868 }
869 }
870 }
871
872 if( finished )
873 break; // only the first tool gets the event
874 }
875
876 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::dispatchInternal - %s handle event: %s" ),
877 ( handled ? wxS( "Did" ) : wxS( "Did not" ) ), aEvent.Format() );
878
879 return handled;
880}
881
882
884{
885 if( aEvent.Action() == TA_KEY_PRESSED )
886 return m_actionMgr->RunHotKey( aEvent.Modifier() | aEvent.KeyCode() );
887
888 return false;
889}
890
891
893{
894 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::dispatchActivation - Received event: %s" ),
895 aEvent.Format() );
896
897 if( aEvent.IsActivate() )
898 {
899 auto tool = m_toolNameIndex.find( aEvent.getCommandStr() );
900
901 if( tool != m_toolNameIndex.end() )
902 {
903 wxLogTrace( kicadTraceToolStack,
904 wxS( "TOOL_MANAGER::dispatchActivation - Running tool %s for event: %s" ),
905 tool->second->theTool->GetName(), aEvent.Format() );
906
907 runTool( tool->second->theTool );
908 return true;
909 }
910 }
911
912 return false;
913}
914
915
917{
918 // Don't open context menus if we're inside a yielding event loop such as a progress dialog.
919 // Opening a popup menu during YieldFor creates a nested modal situation that can leave the
920 // menu stuck and unresponsive, potentially locking up the entire UI on some platforms.
921 if( wxEventLoopBase* loop = wxEventLoopBase::GetActive() )
922 {
923 if( loop->IsYielding() )
924 return;
925 }
926
927 for( TOOL_ID toolId : m_activeTools )
928 {
929 TOOL_STATE* st = m_toolIdIndex[toolId];
930
931 // the tool requested a context menu. The menu is activated on RMB click (CMENU_BUTTON mode)
932 // or immediately (CMENU_NOW) mode. The latter is used for clarification lists.
933 if( st->contextMenuTrigger == CMENU_OFF )
934 continue;
935
936 if( st->contextMenuTrigger == CMENU_BUTTON && !aEvent.IsClick( BUT_RIGHT ) )
937 break;
938
939 if( st->cofunc )
940 {
941 st->pendingWait = true;
943 }
944
945 // Store the menu pointer in case it is changed by the TOOL when handling menu events
946 ACTION_MENU* m = st->contextMenu;
947
948 if( st->contextMenuTrigger == CMENU_NOW )
950
951 // Store the cursor position, so the tools could execute actions
952 // using the point where the user has invoked a context menu
953 if( m_viewControls )
954 m_menuCursor = m_viewControls->GetCursorPosition();
955
956 // Save all tools cursor settings, as they will be overridden
957 for( const std::pair<const TOOL_ID, TOOL_STATE*>& idState : m_toolIdIndex )
958 {
959 TOOL_STATE* s = idState.second;
960 const auto& vc = s->vcSettings;
961
962 if( vc.m_forceCursorPosition )
963 m_cursorSettings[idState.first] = vc.m_forcedPosition;
964 else
965 m_cursorSettings[idState.first] = std::nullopt;
966 }
967
968 if( m_viewControls )
969 m_viewControls->ForceCursorPosition( true, m_menuCursor );
970
971 // Display a copy of menu
972 std::unique_ptr<ACTION_MENU> menu( m->Clone() );
973
974 m_menuOwner = toolId;
975 m_menuActive = true;
976
977 if( wxWindow* frame = dynamic_cast<wxWindow*>( m_frame ) )
978 frame->PopupMenu( menu.get() );
979
980 // Warp the cursor if a menu item was selected
981 if( menu->GetSelected() >= 0 )
982 {
984 m_viewControls->WarpMouseCursor( m_menuCursor, true, false );
985 }
986 // Otherwise notify the tool of a canceled menu
987 else
988 {
990 evt.SetHasPosition( false );
991 evt.SetParameter( m );
992 dispatchInternal( evt );
993 }
994
995 // Restore setting in case it was vetoed
997
998 // Notify the tools that menu has been closed
1000 evt.SetHasPosition( false );
1001 evt.SetParameter( m );
1002 dispatchInternal( evt );
1003
1004 m_menuActive = false;
1005 m_menuOwner = -1;
1006
1007 // Restore cursor settings
1008 for( const std::pair<const TOOL_ID,
1009 std::optional<VECTOR2D>>& cursorSetting : m_cursorSettings )
1010 {
1011 auto it = m_toolIdIndex.find( cursorSetting.first );
1012 wxASSERT( it != m_toolIdIndex.end() );
1013
1014 if( it == m_toolIdIndex.end() )
1015 continue;
1016
1017 KIGFX::VC_SETTINGS& vc = it->second->vcSettings;
1018 vc.m_forceCursorPosition = (bool) cursorSetting.second;
1019 vc.m_forcedPosition = cursorSetting.second ? *cursorSetting.second : VECTOR2D( 0, 0 );
1020 }
1021
1022 m_cursorSettings.clear();
1023 break;
1024 }
1025}
1026
1027
1029{
1031 m_viewControls->WarpMouseCursor( m_menuCursor, true, false );
1032
1033 // Don't warp again when the menu is closed
1035}
1036
1037
1038TOOL_MANAGER::ID_LIST::iterator TOOL_MANAGER::finishTool( TOOL_STATE* aState )
1039{
1040 auto it = std::find( m_activeTools.begin(), m_activeTools.end(), aState->theTool->GetId() );
1041
1042 if( !aState->Pop() )
1043 {
1044 // Deactivate the tool if there are no other contexts saved on the stack
1045 if( it != m_activeTools.end() )
1046 it = m_activeTools.erase( it );
1047
1048 aState->idle = true;
1049 }
1050
1051 if( aState == m_activeState )
1052 setActiveState( nullptr );
1053
1054 return it;
1055}
1056
1057
1059{
1060 // Once the tool manager is shutting down, don't start
1061 // activating more tools
1062 if( m_shuttingDown )
1063 return true;
1064
1065 bool handled = processEvent( aEvent );
1066
1067 TOOL_STATE* activeTool = GetCurrentToolState();
1068
1069 if( activeTool )
1070 setActiveState( activeTool );
1071
1072 if( m_view && m_view->IsDirty() )
1073 {
1074#if defined( __WXMAC__ )
1075 wxTheApp->ProcessPendingEvents(); // required for updating brightening behind a popup menu
1076#endif
1077 }
1078
1079 UpdateUI( aEvent );
1080
1081 return handled;
1082}
1083
1084
1086 CONTEXT_MENU_TRIGGER aTrigger )
1087{
1088 TOOL_STATE* st = m_toolState[aTool];
1089
1090 st->contextMenu = aMenu;
1091 st->contextMenuTrigger = aTrigger;
1092}
1093
1094
1096{
1097 if( TOOL_STATE* active = GetCurrentToolState() )
1098 return active->vcSettings;
1099
1100 return m_viewControls->GetSettings();
1101}
1102
1103
1104TOOL_ID TOOL_MANAGER::MakeToolId( const std::string& aToolName )
1105{
1106 static int currentId;
1107
1108 return currentId++;
1109}
1110
1111
1113 KIGFX::VIEW_CONTROLS* aViewControls,
1114 APP_SETTINGS_BASE* aSettings, TOOLS_HOLDER* aFrame )
1115{
1116 m_model = aModel;
1117 m_view = aView;
1118 m_viewControls = aViewControls;
1119 m_frame = aFrame;
1120 m_settings = aSettings;
1121}
1122
1123
1125{
1126 if( !isRegistered( aTool ) )
1127 return false;
1128
1129 // Just check if the tool is on the active tools stack
1130 return alg::contains( m_activeTools, aTool->GetId() );
1131}
1132
1133
1135{
1136 aState->vcSettings = m_viewControls->GetSettings();
1137
1138 if( m_menuActive )
1139 {
1140 // Context menu is active, so the cursor settings are overridden (see DispatchContextMenu())
1141 auto it = m_cursorSettings.find( aState->theTool->GetId() );
1142
1143 if( it != m_cursorSettings.end() )
1144 {
1145 const KIGFX::VC_SETTINGS& curr = m_viewControls->GetSettings();
1146
1147 // Tool has overridden the cursor position, so store the new settings
1149 {
1150 if( !curr.m_forceCursorPosition )
1151 it->second = std::nullopt;
1152 else
1153 it->second = curr.m_forcedPosition;
1154 }
1155 else
1156 {
1157 std::optional<VECTOR2D> cursor = it->second;
1158
1159 if( cursor )
1160 {
1161 aState->vcSettings.m_forceCursorPosition = true;
1163 }
1164 else
1165 {
1166 aState->vcSettings.m_forceCursorPosition = false;
1167 }
1168 }
1169 }
1170 }
1171}
1172
1173
1175{
1176 m_viewControls->ApplySettings( aState->vcSettings );
1177}
1178
1179
1181{
1182 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::processEvent - %s" ), aEvent.Format() );
1183
1184 // Capture the cursor position from the keyboard event so that hotkey-triggered actions use
1185 // the position at keypress time rather than polling a potentially stale position later in the
1186 // dispatch chain. The scoped guard restores any prior value so a nested hotkey dispatch does
1187 // not clobber the outer position.
1188 std::optional<VECTOR2D> hotKeyPos = aEvent.HasPosition() && aEvent.Action() == TA_KEY_PRESSED
1189 ? std::make_optional( aEvent.Position() )
1190 : m_hotKeyPos;
1191
1192 SCOPED_SET_RESET<std::optional<VECTOR2D>> scopedHotKeyPos( m_hotKeyPos, hotKeyPos );
1193
1194 // First try to dispatch the action associated with the event if it is a key press event
1195 bool handled = DispatchHotKey( aEvent );
1196
1197 if( !handled )
1198 {
1199 TOOL_EVENT mod_event( aEvent );
1200
1201 // Only immediate actions get the position. Otherwise clear for tool activation
1202 if( GetToolHolder() && !GetToolHolder()->GetDoImmediateActions() )
1203 {
1204 // An tool-selection-event has no position
1205 if( !mod_event.getCommandStr().empty()
1206 && mod_event.getCommandStr() != GetToolHolder()->CurrentToolName()
1207 && !mod_event.ForceImmediate() )
1208 {
1209 mod_event.SetHasPosition( false );
1210 }
1211 }
1212
1213 // If the event is not handled through a hotkey activation, pass it to the currently
1214 // running tool loops
1215 handled |= dispatchInternal( mod_event );
1216 handled |= dispatchActivation( mod_event );
1217
1218 // Open the context menu if requested by a tool
1219 DispatchContextMenu( mod_event );
1220
1221 // Dispatch any remaining events in the event queue
1222 while( !m_eventQueue.empty() )
1223 {
1224 TOOL_EVENT event = m_eventQueue.front();
1225 m_eventQueue.pop_front();
1226 processEvent( event );
1227 }
1228 }
1229
1230 wxLogTrace( kicadTraceToolStack, wxS( "TOOL_MANAGER::processEvent - %s handle event: %s" ),
1231 ( handled ? "Did" : "Did not" ), aEvent.Format() );
1232
1233 return handled;
1234}
1235
1236
1238{
1241
1242 m_activeState = aState;
1243
1245 applyViewControls( aState );
1246}
1247
1248
1250{
1251 auto it = m_toolIdIndex.find( aId );
1252
1253 if( it == m_toolIdIndex.end() )
1254 return false;
1255
1256 return !it->second->idle;
1257}
1258
1259
1261{
1262 EDA_BASE_FRAME* frame = dynamic_cast<EDA_BASE_FRAME*>( GetToolHolder() );
1263
1264 if( frame )
1265 frame->UpdateStatusBar();
1266}
Manage TOOL_ACTION objects.
Define the structure of a menu based on ACTIONs.
Definition action_menu.h:47
ACTION_MENU * Clone() const
Create a deep, recursive copy of this ACTION_MENU.
APP_SETTINGS_BASE is a settings class that should be derived for each standalone KiCad application.
Represent a set of changes (additions, deletions or modifications) of a data model (e....
Definition commit.h:72
Implement a coroutine.
Definition coroutine.h:84
bool Call(ArgType aArg)
Start execution of a coroutine, passing args as its arguments.
Definition coroutine.h:286
void KiYield()
Stop execution of the coroutine and returns control to the caller.
Definition coroutine.h:249
bool Resume()
Resume execution of a previously yielded coroutine.
Definition coroutine.h:332
bool Running() const
Definition coroutine.h:381
void RunMainStack(std::function< void()> func)
Run a functor inside the application main stack context.
Definition coroutine.h:272
The base frame for deriving all KiCad main window classes.
virtual void UpdateStatusBar()
Update the status bar information.
A base class for most all the KiCad significant classes used in schematics and boards.
Definition eda_item.h:100
An interface for classes handling user events controlling the view behavior such as zooming,...
Hold a (potentially large) number of VIEW_ITEMs and renders them on a graphics device provided by the...
Definition view.h:67
RAII class that sets an value at construction and resets it to the original value at destruction.
Represent a single user action.
TOOL_EVENT MakeEvent() const
Return the event associated with the action (i.e.
Base abstract interface for all kinds of tools.
Definition tool_base.h:66
virtual void Reset(RESET_REASON aReason)=0
Bring the tool to a known, initial state.
virtual bool Init()
Init() is called once upon a registration of the tool.
Definition tool_base.h:92
const std::string & GetName() const
Return the name of the tool.
Definition tool_base.h:136
TOOL_TYPE GetType() const
Return the type of the tool.
Definition tool_base.h:111
RESET_REASON
Determine the reason of reset for a tool.
Definition tool_base.h:78
@ REDRAW
Full drawing refresh.
Definition tool_base.h:83
@ RUN
Tool is invoked after being inactive.
Definition tool_base.h:79
TOOL_ID GetId() const
Return the unique identifier of the tool.
Definition tool_base.h:123
void attachManager(TOOL_MANAGER *aManager)
Set the TOOL_MANAGER the tool will belong to.
Definition tool_base.cpp:62
A list of TOOL_EVENTs, with overloaded || operators allowing for concatenating TOOL_EVENTs with littl...
Definition tool_event.h:648
OPT_TOOL_EVENT Matches(const TOOL_EVENT &aEvent) const
Definition tool_event.h:687
Generic, UI-independent tool event.
Definition tool_event.h:171
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:260
bool PassEvent() const
These give a tool a method of informing the TOOL_MANAGER that a particular event should be passed on ...
Definition tool_event.h:255
TOOL_ACTIONS Action() const
Returns more specific information about the type of an event.
Definition tool_event.h:250
void SetMousePosition(const VECTOR2D &aP)
Definition tool_event.h:538
int KeyCode() const
Definition tool_event.h:376
TOOL_BASE * FirstResponder() const
Definition tool_event.h:268
bool IsActivate() const
Definition tool_event.h:345
void SetFirstResponder(TOOL_BASE *aTool)
Definition tool_event.h:269
bool IsSimulator() const
Indicate if the event is from the simulator.
const VECTOR2D Position() const
Return mouse cursor position in world coordinates.
Definition tool_event.h:293
void SetParameter(T aParam)
Set a non-standard parameter assigned to the event.
Definition tool_event.h:528
bool ForceImmediate() const
Returns if the action associated with this event should be treated as immediate regardless of the cur...
Definition tool_event.h:265
bool IsClick(int aButtonMask=BUT_ANY) const
TOOL_EVENT_CATEGORY Category() const
Return the category (eg. mouse/keyboard/action) of an event.
Definition tool_event.h:247
int Modifier(int aMask=MD_MODIFIER_MASK) const
Return information about key modifiers state (Ctrl, Alt, etc.).
Definition tool_event.h:366
void SetHasPosition(bool aHasPosition)
Definition tool_event.h:261
const std::string & getCommandStr() const
Definition tool_event.h:558
bool IsChoiceMenu() const
Definition tool_event.h:355
const std::string Format() const
Return information about event in form of a human-readable string.
void applyViewControls(const TOOL_STATE *aState)
Apply #VIEW_CONTROLS settings stored in a TOOL_STATE object.
int GetPriority(int aToolId) const
Return priority of a given tool.
bool ProcessEvent(const TOOL_EVENT &aEvent)
Propagate an event to tools that requested events of matching type(s).
TOOL_STATE * m_activeState
Pointer to the state object corresponding to the currently executed tool.
void UpdateUI(const TOOL_EVENT &aEvent)
Update the status bar and synchronizes toolbars.
std::map< TOOL_ID, std::optional< VECTOR2D > > m_cursorSettings
Original cursor position, if overridden by the context menu handler.
void PostEvent(const TOOL_EVENT &aEvent)
Put an event to the event queue to be processed at the end of event processing cycle.
void ScheduleNextState(TOOL_BASE *aTool, TOOL_STATE_FUNC &aHandler, const TOOL_EVENT_LIST &aConditions)
Define a state transition.
std::map< const char *, TOOL_BASE * > m_toolTypes
Index of the registered tools to easily lookup by their type.
bool isRegistered(TOOL_BASE *aTool) const
Return information about a tool registration status.
APP_SETTINGS_BASE * m_settings
TOOL_STATE * GetCurrentToolState() const
Return the TOOL_STATE object representing the state of the active tool.
VECTOR2D GetCursorPosition() const
std::list< TOOL_EVENT > m_eventQueue
Queue that stores events to be processed at the end of the event processing cycle.
std::pair< TOOL_EVENT_LIST, TOOL_STATE_FUNC > TRANSITION
void setActiveState(TOOL_STATE *aState)
Save the previous active state and sets a new one.
void DeactivateTool()
Deactivate the currently active tool.
std::optional< VECTOR2D > m_hotKeyPos
Mouse position captured at hotkey time, used to avoid the delay between keypress and action dispatch ...
bool dispatchInternal(TOOL_EVENT &aEvent)
Pass an event at first to the active tools, then to all others.
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...
bool runTool(TOOL_BASE *aTool)
Make a tool active, so it can receive events and react to them.
bool InvokeTool(TOOL_ID aToolId)
Call a tool by sending a tool activation event to tool of given ID.
NAME_STATE_MAP m_toolNameIndex
Index of the registered tools current states, associated by tools' names.
TOOLS_HOLDER * m_frame
void CancelTool()
Send a cancel event to the tool currently at the top of the tool stack.
bool m_warpMouseAfterContextMenu
KIGFX::VIEW_CONTROLS * m_viewControls
void WarpAfterContextMenu()
Normally we warp the mouse after the context menu action runs.
ACTION_MANAGER * m_actionMgr
Instance of ACTION_MANAGER that handles TOOL_ACTIONs.
const KIGFX::VC_SETTINGS & GetCurrentToolVC() const
Return the view controls settings for the current tool or the general settings if there is no active ...
void RunMainStack(TOOL_BASE *aTool, std::function< void()> aFunc)
TOOLS_HOLDER * GetToolHolder() const
VECTOR2D GetMousePosition() const
bool processEvent(const TOOL_EVENT &aEvent)
Main function for event processing.
std::list< TOOL_ID > ID_LIST
ID_LIST::iterator finishTool(TOOL_STATE *aState)
Deactivate a tool and does the necessary clean up.
bool doRunAction(const TOOL_ACTION &aAction, bool aNow, const ki::any &aParam, COMMIT *aCommit, bool aFromAPI=false)
Helper function to actually run an action.
bool dispatchActivation(const TOOL_EVENT &aEvent)
Check if it is a valid activation event and invokes a proper tool.
ID_STATE_MAP m_toolIdIndex
Index of the registered tools current states, associated by tools' ID numbers.
TOOL_ID m_menuOwner
Tool currently displaying a popup menu. It is negative when there is no menu displayed.
KIGFX::VIEW * m_view
void ClearTransitions(TOOL_BASE *aTool)
Clear the state transition map for a tool.
bool m_shuttingDown
True if the tool manager is shutting down (don't process additional events)
void saveViewControls(TOOL_STATE *aState)
Save the #VIEW_CONTROLS settings to the tool state object.
int GetHotKey(const TOOL_ACTION &aAction) const
Return the hot key associated with a given action or 0 if there is none.
bool m_menuActive
Flag indicating whether a context menu is currently displayed.
TOOL_STATE_MAP m_toolState
Index of registered tools current states, associated by tools' objects.
void ShutdownTool(TOOL_BASE *aTool)
Shutdown the specified tool by waking it up with a null event to terminate the processing loop.
ID_LIST m_activeTools
Stack of the active tools.
void ResetTools(TOOL_BASE::RESET_REASON aReason)
Reset all tools (i.e.
EDA_ITEM * m_model
bool DispatchHotKey(const TOOL_EVENT &aEvent)
Handle specific events, that are intended for TOOL_MANAGER rather than tools.
static TOOL_ID MakeToolId(const std::string &aToolName)
Generate a unique ID from for a tool with given name.
bool invokeTool(TOOL_BASE *aTool)
Invoke a tool by sending a proper event (in contrary to runTool, which makes the tool run for real).
void DispatchContextMenu(const TOOL_EVENT &aEvent)
Handle context menu related events.
std::vector< TOOL_BASE * > m_toolOrder
List of tools in the order they were registered.
TOOL_BASE * FindTool(int aId) const
Search for a tool with given ID.
void ScheduleContextMenu(TOOL_BASE *aTool, ACTION_MENU *aMenu, CONTEXT_MENU_TRIGGER aTrigger)
Set behavior of the tool's context popup menu.
TOOL_EVENT * ScheduleWait(TOOL_BASE *aTool, const TOOL_EVENT_LIST &aConditions)
Pause execution of a given tool until one or more events matching aConditions arrives.
bool IsToolActive(TOOL_ID aId) const
Return true if a tool with given id is active (executing)
bool isActive(TOOL_BASE *aTool) const
Return information about a tool activation status.
void RegisterTool(TOOL_BASE *aTool)
Add a tool to the manager set and sets it up.
void SetEnvironment(EDA_ITEM *aModel, KIGFX::VIEW *aView, KIGFX::VIEW_CONTROLS *aViewControls, APP_SETTINGS_BASE *aSettings, TOOLS_HOLDER *aFrame)
Set the work environment (model, view, view controls and the parent window).
void InitTools()
Initialize all registered tools.
VECTOR2D m_menuCursor
Right click context menu position.
void ShutdownAllTools()
Shutdown all tools with a currently registered event loop in this tool manager by waking them up with...
A type-safe container of any type.
Definition ki_any.h:93
bool has_value() const noexcept
Report whether there is a contained object or not.
Definition ki_any.h:312
Base window classes and related definitions.
const wxChar *const kicadTraceToolStack
Flag to enable tracing of the tool handling stack.
void AddTransactionBreadcrumb(const wxString &aMsg, const wxString &aCategory)
Add a transaction breadcrumb.
wxPoint GetMousePosition()
Returns the mouse position in screen coordinates.
Definition wxgtk/ui.cpp:766
bool contains(const _Container &__container, _Value __value)
Returns true if the container contains the given value.
Definition kicad_algo.h:100
Structure to keep VIEW_CONTROLS settings for easy store/restore operations.
VECTOR2D m_forcedPosition
Forced cursor position (world coordinates).
bool m_forceCursorPosition
Is the forced cursor position enabled.
Struct describing the current execution state of a TOOL.
bool operator!=(const TOOL_MANAGER::TOOL_STATE &aRhs) const
bool Pop()
Restore state of the tool from stack.
bool pendingWait
Flag defining if the tool is waiting for any event (i.e.
bool pendingContextMenu
Is there a context menu being displayed.
void Push()
Store the current state of the tool on stack.
CONTEXT_MENU_TRIGGER contextMenuTrigger
Defines when the context menu is opened.
void resetRuntimeState()
Resets runtime-only state that must not leak across tool activations.
TOOL_EVENT initialEvent
The first event that triggered activation of the tool.
KIGFX::VC_SETTINGS vcSettings
VIEW_CONTROLS settings to preserve settings when the tools are switched.
TOOL_STATE(const TOOL_STATE &aState)
TOOL_STATE(TOOL_BASE *aTool)
ACTION_MENU * contextMenu
Context menu currently used by the tool.
bool idle
Is the tool active (pending execution) or disabled at the moment.
std::vector< TRANSITION > transitions
List of possible transitions (ie.
void clear()
Restores the initial state.
TOOL_EVENT_LIST waitEvents
List of events the tool is currently waiting for.
COROUTINE< int, const TOOL_EVENT & > * cofunc
Tool execution context.
bool shutdown
Should the tool shutdown during next execution.
std::stack< std::unique_ptr< TOOL_STATE > > stateStack
Stack preserving previous states of a TOOL.
TOOL_BASE * theTool
The tool itself.
TOOL_EVENT wakeupEvent
The event that triggered the execution/wakeup of the tool after Wait() call.
TOOL_STATE & operator=(const TOOL_STATE &aState)
bool operator==(const TOOL_MANAGER::TOOL_STATE &aRhs) const
VECTOR2I end
std::function< int(const TOOL_EVENT &)> TOOL_STATE_FUNC
Definition tool_base.h:58
int TOOL_ID
Unique identifier for tools.
Definition tool_base.h:56
@ INTERACTIVE
Tool that interacts with the user.
Definition tool_base.h:49
@ TA_ANY
Definition tool_event.h:126
@ TA_CHOICE_MENU_CHOICE
Context menu choice.
Definition tool_event.h:98
@ TA_ACTIVATE
Tool activation event.
Definition tool_event.h:115
@ TA_CHOICE_MENU_CLOSED
Context menu is closed, no matter whether anything has been chosen or not.
Definition tool_event.h:101
@ TA_PRIME
Tool priming event (a special mouse click).
Definition tool_event.h:124
@ TA_KEY_PRESSED
Definition tool_event.h:76
@ TA_CANCEL_TOOL
Tool cancel event.
Definition tool_event.h:90
CONTEXT_MENU_TRIGGER
Defines when a context menu is opened.
Definition tool_event.h:154
@ CMENU_NOW
Right now (after TOOL_INTERACTIVE::SetContextMenu).
Definition tool_event.h:156
@ CMENU_OFF
Never.
Definition tool_event.h:157
@ CMENU_BUTTON
On the right button.
Definition tool_event.h:155
@ STS_CANCELLED
Definition tool_event.h:164
@ STS_FINISHED
Definition tool_event.h:163
@ STS_RUNNING
Definition tool_event.h:162
@ TC_ANY
Definition tool_event.h:60
@ TC_COMMAND
Definition tool_event.h:57
@ TC_MOUSE
Definition tool_event.h:55
@ TC_MESSAGE
Definition tool_event.h:58
@ BUT_LEFT
Definition tool_event.h:132
@ BUT_RIGHT
Definition tool_event.h:133
wxLogTrace helper definitions.
VECTOR2< double > VECTOR2D
Definition vector2d.h:686
VECTOR2D ToVECTOR2D(const wxPoint &aPoint)
Definition vector2wx.h:40