#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 #include #include // boost::polymorphic_downcast #include 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