ossrv_pub/boost_apis/boost/statechart/transition.hpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 02 Feb 2010 02:01:42 +0200
changeset 0 e4d67989cc36
permissions -rw-r--r--
Revision: 201002 Kit: 201005

#ifndef BOOST_STATECHART_TRANSITION_HPP_INCLUDED
#define BOOST_STATECHART_TRANSITION_HPP_INCLUDED
//////////////////////////////////////////////////////////////////////////////
// Copyright 2002-2006 Andreas Huber Doenni
// Distributed under the Boost Software License, Version 1.0. (See accompany-
// ing file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
//////////////////////////////////////////////////////////////////////////////



#include <boost/statechart/result.hpp>

#include <boost/mpl/if.hpp>

#include <boost/cast.hpp> // boost::polymorphic_downcast
#include <boost/type_traits/is_same.hpp>



namespace boost
{
namespace statechart
{
namespace detail
{



//////////////////////////////////////////////////////////////////////////////
template< class Event >
struct no_context
{
  void no_function( const Event & );
};



} // namespace detail



class event_base;

//////////////////////////////////////////////////////////////////////////////
template< class Event, class Destination,
          class TransitionContext = detail::no_context< Event >,
          void ( TransitionContext::*pTransitionAction )( const Event & ) =
            &detail::no_context< Event >::no_function >
class transition
{
  private:
    //////////////////////////////////////////////////////////////////////////
    struct react_without_transition_action_impl
    {
      template< class State, class EventBase >
      static detail::reaction_result react( State & stt, const EventBase & )
      {
        return detail::result_utility::get_result(
          stt.template transit< Destination >() );
      }
    };

    struct react_base_with_transition_action_impl
    {
      template< class State, class EventBase >
      static detail::reaction_result react(
        State & stt, const EventBase & toEvent )
      {
        return detail::result_utility::get_result(
          stt.template transit< Destination >( pTransitionAction, toEvent ) );
      }
    };

    struct react_base
    {
      template< class State, class EventBase, class IdType >
      static detail::reaction_result react(
        State & stt, const EventBase & evt, const IdType & )
      {
        typedef typename mpl::if_<
          is_same< TransitionContext, detail::no_context< Event > >,
          react_without_transition_action_impl,
          react_base_with_transition_action_impl
        >::type impl;
        return impl::react( stt, evt );
      }
    };

    struct react_derived_with_transition_action_impl
    {
      template< class State, class EventBase >
      static detail::reaction_result react(
        State & stt, const EventBase & toEvent )
      {
        return detail::result_utility::get_result(
          stt.template transit< Destination >(
            pTransitionAction,
            *polymorphic_downcast< const Event * >( &toEvent ) ) );
      }
    };

    struct react_derived
    {
      template< class State, class EventBase, class IdType >
      static detail::reaction_result react(
        State & stt, const EventBase & evt, const IdType & eventType )
      {
        if ( eventType == Event::static_type() )
        {
          typedef typename mpl::if_<
            is_same< TransitionContext, detail::no_context< Event > >,
            react_without_transition_action_impl,
            react_derived_with_transition_action_impl
          >::type impl;
          return impl::react( stt, evt );
        }
        else
        {
          return detail::no_reaction;
        }
      }
    };

  public:
    //////////////////////////////////////////////////////////////////////////
    // The following declarations should be private.
    // They are only public because many compilers lack template friends.
    //////////////////////////////////////////////////////////////////////////
    template< class State, class EventBase, class IdType >
    static detail::reaction_result react(
      State & stt, const EventBase & evt, const IdType & eventType )
    {
      typedef typename mpl::if_<
        is_same< Event, event_base >, react_base, react_derived
      >::type impl;

      return impl::react( stt, evt, eventType );
    }
};



} // namespace statechart
} // namespace boost



#endif