/* * Copyright (C) 2015 Naios * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef _CONTINUABLE_H_ #define _CONTINUABLE_H_ #include "Callback.h" namespace detail { /// The internal state of the continuable /// which is used to save certain internal types. template struct ContinuableState; template struct ContinuableState, _Proxy> { }; typedef ContinuableState, void> DefaultContinuableState; // ContinuableImpl Forward definition template class _ContinuableImpl; /// Corrects void return types from functional types which should be Continuable> template struct convert_void_to_continuable; template<> struct convert_void_to_continuable { typedef _ContinuableImpl> type; }; template struct convert_void_to_continuable<_ContinuableImpl<_State, _CTy>> { typedef _ContinuableImpl<_State, _CTy> type; }; template class _ContinuableImpl, std::function> { // Make all instances of _ContinuableImpl to a friend. template friend class _ContinuableImpl; public: typedef std::function&&)> ForwardFunction; private: /// Functional which expects a callback that is inserted from the Continuable /// to chain everything together ForwardFunction _callback_insert; bool _released; void Dispatch() { } public: // Empty for debugging _ContinuableImpl() : _released(false), _callback_insert() { } /// Deleted copy construct _ContinuableImpl(_ContinuableImpl const&) = delete; /// Move construct _ContinuableImpl(_ContinuableImpl&& right) : _released(right._released), _callback_insert(std::move(right._callback_insert)) { right._released = true; } // Construct through a ForwardFunction template _ContinuableImpl(_FTy&& callback_insert) : _callback_insert(std::forward<_FTy>(callback_insert)), _released(false) { } /// Destructor which calls the dispatch chain if needed. ~_ContinuableImpl() { if (!_released) { _released = true; Dispatch(); } } /// Deleted copy assign template _ContinuableImpl& operator= (_ContinuableImpl const&) = delete; /// Move construct assign _ContinuableImpl& operator= (_ContinuableImpl&& right) { _released = right._released; right._released = true; _callback_insert = std::move(right._callback_insert); return *this; } template struct unary_chainer; template struct unary_chainer> { template static auto chain(_CTy&& functional, _ContinuableImpl&& me) -> typename convert_void_to_continuable<_NewRTy>::type { return typename convert_void_to_continuable<_NewRTy>::type(); // _ContinuableImpl, Callback<_NewATy...>>(); } }; template auto then(_CTy&& functional) -> decltype(unary_chainer::type>>:: chain(std::declval<_CTy>(), std::declval<_ContinuableImpl&&>())) { return unary_chainer::type>>:: chain(std::forward<_CTy>(functional), std::move(*this)); } /* // TODO Accept only correct callbacks template Continuable> all(_CTy&&...) { // TODO Transmute the returned callback here. return Continuable>(std::move(*this)); } */ /// Invalidates the Continuable _ContinuableImpl& invalidate() { _released = true; return *this; } }; } /// A continuable provides useful methods to react on the result of callbacks /// and allows to chain multiple callback calls to a chain. template using Continuable = detail::_ContinuableImpl< detail::DefaultContinuableState, Callback>; namespace detail { template struct ContinuableFactory; template struct ContinuableFactory<_RTy, ::fu::identity&&>> { template static auto CreateFrom(_FTy&& functional) -> Continuable<_ATy...> { return Continuable<_ATy...>( typename Continuable<_ATy...>::ForwardFunction(std::forward<_FTy>(functional))); } }; template using continuable_factory_t = ContinuableFactory< ::fu::return_type_of_t<_FTy>, ::fu::argument_type_of_t<_FTy>>; } // detail /// Wraps a functional object which expects a r-value callback as argument into a continuable. /// The callable is invoked when the continuable shall continue. /// For example: /// make_continuable([](Callback&& callback) /// { /// /* Continue here */ /// callback(5); /// }); template inline auto make_continuable(_FTy&& functional) -> decltype(detail::continuable_factory_t<_FTy>::CreateFrom(std::declval<_FTy>())) { return detail::continuable_factory_t<_FTy>::CreateFrom(std::forward<_FTy>(functional)); } #endif // _CONTINUABLE_H_