Tool Manager: yet another approach to handling tool view control settings

All calls to {save,apply}ViewControls() have been replaced with a new
method: setActiveState(). The advantage is that setActiveState() always
saves view controls settings to the tool that set them. As long
as setActiveState() is called every time there is a tool context
switch, the changes are kept up-to-date.

Fixes: lp:1748613
* https://bugs.launchpad.net/kicad/+bug/1748613
This commit is contained in:
Maciej Suminski 2018-02-14 15:09:10 +01:00
parent 21a2c8d562
commit 151021919e
2 changed files with 46 additions and 23 deletions

View File

@ -297,10 +297,16 @@ void TOOL_MANAGER::RunAction( const TOOL_ACTION& aAction, bool aNow, void* aPara
event.SetParameter( aParam ); event.SetParameter( aParam );
if( aNow ) if( aNow )
{
TOOL_STATE* current = m_activeState;
processEvent( event ); processEvent( event );
setActiveState( current );
}
else else
{
PostEvent( event ); PostEvent( event );
} }
}
int TOOL_MANAGER::GetHotKey( const TOOL_ACTION& aAction ) int TOOL_MANAGER::GetHotKey( const TOOL_ACTION& aAction )
@ -322,6 +328,9 @@ bool TOOL_MANAGER::invokeTool( TOOL_BASE* aTool )
TOOL_EVENT evt( TC_COMMAND, TA_ACTIVATE, aTool->GetName() ); TOOL_EVENT evt( TC_COMMAND, TA_ACTIVATE, aTool->GetName() );
processEvent( evt ); processEvent( evt );
if( TOOL_STATE* active = GetCurrentToolState() )
setActiveState( active );
return true; return true;
} }
@ -371,7 +380,7 @@ bool TOOL_MANAGER::runTool( TOOL_BASE* aTool )
return false; return false;
} }
m_activeState = m_toolIdIndex[id]; setActiveState( m_toolIdIndex[id] );
aTool->Reset( TOOL_INTERACTIVE::RUN ); aTool->Reset( TOOL_INTERACTIVE::RUN );
// Add the tool on the front of the processing queue (it gets events first) // Add the tool on the front of the processing queue (it gets events first)
@ -417,8 +426,8 @@ void TOOL_MANAGER::ResetTools( TOOL_BASE::RESET_REASON aReason )
for( auto& state : m_toolState ) for( auto& state : m_toolState )
{ {
m_activeState = state.second;
TOOL_BASE* tool = state.first; TOOL_BASE* tool = state.first;
setActiveState( state.second );
tool->Reset( aReason ); tool->Reset( aReason );
if( tool->GetType() == INTERACTIVE ) if( tool->GetType() == INTERACTIVE )
@ -433,8 +442,8 @@ void TOOL_MANAGER::InitTools()
{ {
TOOL_BASE* tool = it->first; TOOL_BASE* tool = it->first;
TOOL_STATE* state = it->second; TOOL_STATE* state = it->second;
setActiveState( state );
++it; // keep the iterator valid if the element is going to be erased ++it; // keep the iterator valid if the element is going to be erased
m_activeState = state;
if( !tool->Init() ) if( !tool->Init() )
{ {
@ -442,6 +451,7 @@ void TOOL_MANAGER::InitTools()
wxString::Format( "Initialization of tool \"%s\" failed", tool->GetName() ) ); wxString::Format( "Initialization of tool \"%s\" failed", tool->GetName() ) );
// Unregister the tool // Unregister the tool
setActiveState( nullptr );
m_toolState.erase( tool ); m_toolState.erase( tool );
m_toolNameIndex.erase( tool->GetName() ); m_toolNameIndex.erase( tool->GetName() );
m_toolIdIndex.erase( tool->GetId() ); m_toolIdIndex.erase( tool->GetId() );
@ -490,12 +500,12 @@ void TOOL_MANAGER::ClearTransitions( TOOL_BASE* aTool )
void TOOL_MANAGER::RunMainStack( TOOL_BASE* aTool, std::function<void()> aFunc ) void TOOL_MANAGER::RunMainStack( TOOL_BASE* aTool, std::function<void()> aFunc )
{ {
TOOL_STATE* st = m_toolState[aTool]; TOOL_STATE* st = m_toolState[aTool];
setActiveState( st );
st->cofunc->RunMainStack( std::move( aFunc ) ); st->cofunc->RunMainStack( std::move( aFunc ) );
} }
OPT<TOOL_EVENT> TOOL_MANAGER::ScheduleWait( TOOL_BASE* aTool, OPT<TOOL_EVENT> TOOL_MANAGER::ScheduleWait( TOOL_BASE* aTool, const TOOL_EVENT_LIST& aConditions )
const TOOL_EVENT_LIST& aConditions )
{ {
TOOL_STATE* st = m_toolState[aTool]; TOOL_STATE* st = m_toolState[aTool];
@ -515,9 +525,6 @@ OPT<TOOL_EVENT> TOOL_MANAGER::ScheduleWait( TOOL_BASE* aTool,
void TOOL_MANAGER::dispatchInternal( const TOOL_EVENT& aEvent ) void TOOL_MANAGER::dispatchInternal( const TOOL_EVENT& aEvent )
{ {
// Save current settings before overwriting with the dispatched tool
auto vc_settings = m_viewControls->GetSettings();
// iterate over all registered tools // iterate over all registered tools
for( auto it = m_activeTools.begin(); it != m_activeTools.end(); ++it ) for( auto it = m_activeTools.begin(); it != m_activeTools.end(); ++it )
{ {
@ -545,10 +552,8 @@ void TOOL_MANAGER::dispatchInternal( const TOOL_EVENT& aEvent )
if( st->cofunc ) if( st->cofunc )
{ {
applyViewControls( st ); setActiveState( st );
m_activeState = st;
bool end = !st->cofunc->Resume(); bool end = !st->cofunc->Resume();
saveViewControls( st );
if( end ) if( end )
it = finishTool( st ); it = finishTool( st );
@ -577,9 +582,14 @@ void TOOL_MANAGER::dispatchInternal( const TOOL_EVENT& aEvent )
{ {
auto func_copy = tr.second; auto func_copy = tr.second;
// if there is already a context, then store it // if there is already a context, then push it on the stack
// and transfer the previous view control settings to the new context
if( st->cofunc ) if( st->cofunc )
{
auto vc = st->vcSettings;
st->Push(); st->Push();
st->vcSettings = vc;
}
st->cofunc = new COROUTINE<int, const TOOL_EVENT&>( std::move( func_copy ) ); st->cofunc = new COROUTINE<int, const TOOL_EVENT&>( std::move( func_copy ) );
@ -587,10 +597,9 @@ void TOOL_MANAGER::dispatchInternal( const TOOL_EVENT& aEvent )
st->transitions.clear(); st->transitions.clear();
// got match? Run the handler. // got match? Run the handler.
applyViewControls( st ); setActiveState( st );
st->idle = false; st->idle = false;
st->cofunc->Call( aEvent ); st->cofunc->Call( aEvent );
saveViewControls( st );
if( !st->cofunc->Running() ) if( !st->cofunc->Running() )
finishTool( st ); // The couroutine has finished immediately? finishTool( st ); // The couroutine has finished immediately?
@ -607,8 +616,6 @@ void TOOL_MANAGER::dispatchInternal( const TOOL_EVENT& aEvent )
if( finished ) if( finished )
break; // only the first tool gets the event break; // only the first tool gets the event
} }
m_viewControls->ApplySettings( vc_settings );
} }
@ -738,9 +745,6 @@ TOOL_MANAGER::ID_LIST::iterator TOOL_MANAGER::finishTool( TOOL_STATE* aState )
{ {
auto it = std::find( m_activeTools.begin(), m_activeTools.end(), aState->theTool->GetId() ); auto it = std::find( m_activeTools.begin(), m_activeTools.end(), aState->theTool->GetId() );
if( aState == m_activeState )
m_activeState = nullptr;
if( !aState->Pop() ) if( !aState->Pop() )
{ {
// Deactivate the tool if there are no other contexts saved on the stack // Deactivate the tool if there are no other contexts saved on the stack
@ -748,6 +752,9 @@ TOOL_MANAGER::ID_LIST::iterator TOOL_MANAGER::finishTool( TOOL_STATE* aState )
it = m_activeTools.erase( it ); it = m_activeTools.erase( it );
} }
if( aState == m_activeState )
setActiveState( nullptr );
// Set transitions to be ready for future TOOL_EVENTs // Set transitions to be ready for future TOOL_EVENTs
TOOL_BASE* tool = aState->theTool; TOOL_BASE* tool = aState->theTool;
@ -762,13 +769,10 @@ TOOL_MANAGER::ID_LIST::iterator TOOL_MANAGER::finishTool( TOOL_STATE* aState )
bool TOOL_MANAGER::ProcessEvent( const TOOL_EVENT& aEvent ) bool TOOL_MANAGER::ProcessEvent( const TOOL_EVENT& aEvent )
{ {
if( TOOL_STATE* active = GetCurrentToolState() )
saveViewControls( active );
bool hotkey_handled = processEvent( aEvent ); bool hotkey_handled = processEvent( aEvent );
if( TOOL_STATE* active = GetCurrentToolState() ) if( TOOL_STATE* active = GetCurrentToolState() )
applyViewControls( active ); setActiveState( active );
if( m_view->IsDirty() ) if( m_view->IsDirty() )
{ {
@ -928,6 +932,18 @@ bool TOOL_MANAGER::processEvent( const TOOL_EVENT& aEvent )
} }
void TOOL_MANAGER::setActiveState( TOOL_STATE* aState )
{
if( m_activeState )
saveViewControls( m_activeState );
m_activeState = aState;
if( m_activeState )
applyViewControls( aState );
}
bool TOOL_MANAGER::IsToolActive( TOOL_ID aId ) const bool TOOL_MANAGER::IsToolActive( TOOL_ID aId ) const
{ {
auto it = m_toolIdIndex.find( aId ); auto it = m_toolIdIndex.find( aId );

View File

@ -483,6 +483,13 @@ private:
///> @return true if a hotkey was handled ///> @return true if a hotkey was handled
bool processEvent( const TOOL_EVENT& aEvent ); bool processEvent( const TOOL_EVENT& aEvent );
/**
* Saves the previous active state and sets a new one.
* @param aState is the new active state. Might be null to indicate there is no new
* active state.
*/
void setActiveState( TOOL_STATE* aState );
/// Index of registered tools current states, associated by tools' objects. /// Index of registered tools current states, associated by tools' objects.
TOOL_STATE_MAP m_toolState; TOOL_STATE_MAP m_toolState;