All Classes Functions Variables Typedefs Enumerations Enumerator Friends Groups Pages
compose.hpp
1 // The MIT License (MIT)
2 
3 // Copyright (c) 2012-2014 Danny Y., Rapptz
4 
5 // Permission is hereby granted, free of charge, to any person obtaining a copy of
6 // this software and associated documentation files (the "Software"), to deal in
7 // the Software without restriction, including without limitation the rights to
8 // use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
9 // the Software, and to permit persons to whom the Software is furnished to do so,
10 // subject to the following conditions:
11 
12 // The above copyright notice and this permission notice shall be included in all
13 // copies or substantial portions of the Software.
14 
15 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
17 // FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
18 // COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
19 // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
20 // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21 
22 #ifndef GEARS_FUNCTIONAL_COMPOSE_HPP
23 #define GEARS_FUNCTIONAL_COMPOSE_HPP
24 
25 #include <type_traits>
26 #include <utility>
27 
28 namespace gears {
29 namespace functional {
30 namespace detail {
31 template<typename T>
32 using Decayed = typename std::decay<T>::type;
33 
34 template<typename First, typename Second>
35 struct compose_type {
36  First f;
37  Second g;
38  constexpr compose_type(First f, Second g) noexcept: f(std::move(f)), g(std::move(g)) {}
39 
40  template<typename... Args>
41  constexpr auto operator()(Args&&... args) const -> decltype(f(g(std::forward<Args>(args)...))) {
42  return f(g(std::forward<Args>(args)...));
43  }
44 };
45 
46 template<typename F, typename... G>
47 struct composer {
48  using type = compose_type<Decayed<F>, typename composer<G...>::type>;
49 };
50 
51 template<typename F>
52 struct composer<F> {
53  using type = Decayed<F>;
54 };
55 
56 template<typename... Args>
57 using Composer = typename composer<Args...>::type;
58 } // detail
59 
60 template<typename Function>
61 constexpr detail::Decayed<Function> compose(Function&& f) {
62  return std::forward<Function>(f);
63 }
64 
81 template<typename First, typename... Rest>
82 constexpr detail::Composer<First, Rest...> compose(First&& f, Rest&&... args) {
83  return detail::Composer<First, Rest...>(std::forward<First>(f), compose(std::forward<Rest>(args)...));
84 }
85 } // functional
86 } // gears
87 
88 #endif // GEARS_FUNCTIONAL_COMPOSE_HPP