fsm.cpp
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:3k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * fsm.cpp
  3.  *****************************************************************************
  4.  * Copyright (C) 2003 the VideoLAN team
  5.  * $Id: a4db29d0df8e820341bf3ff19a6c0a4d39c04aca $
  6.  *
  7.  * Authors: Cyril Deguet     <asmax@via.ecp.fr>
  8.  *          Olivier Teulière <ipkiss@via.ecp.fr>
  9.  *
  10.  * This program is free software; you can redistribute it and/or modify
  11.  * it under the terms of the GNU General Public License as published by
  12.  * the Free Software Foundation; either version 2 of the License, or
  13.  * (at your option) any later version.
  14.  *
  15.  * This program is distributed in the hope that it will be useful,
  16.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  18.  * GNU General Public License for more details.
  19.  *
  20.  * You should have received a copy of the GNU General Public License
  21.  * along with this program; if not, write to the Free Software
  22.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  23.  *****************************************************************************/
  24. #include "fsm.hpp"
  25. #include "../commands/cmd_generic.hpp"
  26. void FSM::addState( const string &state )
  27. {
  28.     m_states.insert( state );
  29. }
  30. void FSM::addTransition( const string &state1, const string &event,
  31.                          const string &state2, CmdGeneric *pCmd )
  32. {
  33.     // Check that we already know the states
  34.     if( m_states.find( state1 ) == m_states.end() ||
  35.         m_states.find( state2 ) == m_states.end() )
  36.     {
  37.         msg_Warn( getIntf(),
  38.                   "FSM: ignoring transition between invalid states" );
  39.         return;
  40.     }
  41.     Key_t key( state1, event );
  42.     Data_t data( state2, pCmd );
  43.     // Check that the transition doesn't already exist
  44.     if( m_transitions.find( key ) != m_transitions.end() )
  45.     {
  46.         msg_Warn( getIntf(), "FSM: transition already exists" );
  47.         return;
  48.     }
  49.     m_transitions[key] = data;
  50. }
  51. void FSM::setState( const string &state )
  52. {
  53.     if( m_states.find( state ) == m_states.end() )
  54.     {
  55.         msg_Warn( getIntf(), "FSM: trying to set an invalid state" );
  56.         return;
  57.     }
  58.     m_currentState = state;
  59. }
  60. void FSM::handleTransition( const string &event )
  61. {
  62.     string tmpEvent = event;
  63.     Key_t key( m_currentState, event );
  64.     map<Key_t, Data_t>::const_iterator it;
  65.     // Find a transition
  66.     it = m_transitions.find( key );
  67.     // While the matching fails, try to match a more generic transition
  68.     // For example, if "key:up:F" isn't a transition, "key:up" or "key" may be
  69.     while( it == m_transitions.end() &&
  70.            tmpEvent.rfind( ":", tmpEvent.size() ) != string::npos )
  71.     {
  72.         // Cut the last part
  73.         tmpEvent = tmpEvent.substr( 0, tmpEvent.rfind( ":", tmpEvent.size() ) );
  74.         key.second = tmpEvent;
  75.         it = m_transitions.find( key );
  76.     }
  77.     // No transition was found
  78.     if( it == m_transitions.end() )
  79.     {
  80.         return;
  81.     }
  82.     // Change state
  83.     m_currentState = (*it).second.first;
  84.     // Call the callback, if any
  85.     CmdGeneric *pCmd = (*it).second.second;
  86.     if( pCmd != NULL )
  87.     {
  88.         pCmd->execute();
  89.     }
  90. }