Toolbox snapshot
The Reactive C++ Toolbox
Loading...
Searching...
No Matches
Slot.hpp
Go to the documentation of this file.
1// The Reactive C++ Toolbox.
2// Copyright (C) 2013-2019 Swirly Cloud Limited
3// Copyright (C) 2022 Reactive Markets Limited
4//
5// Licensed under the Apache License, Version 2.0 (the "License");
6// you may not use this file except in compliance with the License.
7// You may obtain a copy of the License at
8//
9// http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing, software
12// distributed under the License is distributed on an "AS IS" BASIS,
13// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14// See the License for the specific language governing permissions and
15// limitations under the License.
16
17#ifndef TOOLBOX_UTIL_SLOT_HPP
18#define TOOLBOX_UTIL_SLOT_HPP
19
21
22namespace toolbox {
23inline namespace util {
24
25template <typename Sig>
27
28template <typename RetT, typename... ArgsT>
29class BasicSlot<RetT(ArgsT...)> {
30 public:
31 friend constexpr bool operator==(BasicSlot lhs, BasicSlot rhs) noexcept
32 {
33 return lhs.obj_ == rhs.obj_ && lhs.fn_ == rhs.fn_;
34 }
35 friend constexpr bool operator!=(BasicSlot lhs, BasicSlot rhs) noexcept
36 {
37 return !(lhs == rhs);
38 }
39 constexpr explicit BasicSlot(std::nullptr_t = nullptr) noexcept {}
40 ~BasicSlot() = default;
41
42 // Copy.
43 constexpr BasicSlot(const BasicSlot&) noexcept = default;
44 constexpr BasicSlot& operator=(const BasicSlot&) noexcept = default;
45
46 // Move.
49
50 RetT invoke(ArgsT... args) const { return fn_(obj_, std::forward<ArgsT>(args)...); }
51 RetT operator()(ArgsT... args) const { return fn_(obj_, std::forward<ArgsT>(args)...); }
52 constexpr bool empty() const noexcept { return fn_ == nullptr; }
53 constexpr explicit operator bool() const noexcept { return fn_ != nullptr; }
54
55 // Free function.
56 template <RetT (*FnT)(ArgsT...)>
57 constexpr auto& bind() noexcept
58 {
59 obj_ = nullptr;
60 fn_ = [](void* /*obj*/, ArgsT... args) -> RetT {
61 return FnT(std::forward<ArgsT>(args)...);
62 };
63 return *this;
64 }
65 // Lambda function.
66 template <typename ClassT>
67 constexpr auto& bind(ClassT* obj) noexcept
68 {
69 obj_ = obj;
70 fn_ = [](void* obj, ArgsT... args) -> RetT {
71 return (*static_cast<ClassT*>(obj))(std::forward<ArgsT>(args)...);
72 };
73 return *this;
74 }
75 // Member function.
76 template <auto MemFnT, typename ClassT = typename FunctionTraits<decltype(MemFnT)>::ClassType>
77 constexpr auto& bind(ClassT* obj) noexcept
78 {
79 obj_ = obj;
80 fn_ = [](void* obj, ArgsT... args) -> RetT {
81 return (static_cast<ClassT*>(obj)->*MemFnT)(std::forward<ArgsT>(args)...);
82 };
83 return *this;
84 }
85 void reset(std::nullptr_t = nullptr) noexcept
86 {
87 obj_ = nullptr;
88 fn_ = nullptr;
89 }
90
91 private:
92 void* obj_{nullptr};
93 RetT (*fn_)(void*, ArgsT...){nullptr};
94};
95
96template <auto FnT>
97constexpr auto bind() noexcept
98{
99 using Traits = FunctionTraits<decltype(FnT)>;
100 using Slot = typename Traits::template Apply<BasicSlot>;
101 return Slot{}.template bind<FnT>();
102}
103
104template <typename ClassT>
105constexpr auto bind(ClassT* obj) noexcept
106{
107 using Traits = FunctionTraits<decltype(&ClassT::operator())>;
108 using Slot = typename Traits::template Apply<BasicSlot>;
109 return Slot{}.bind(obj);
110}
111
112template <auto MemFnT, typename ClassT = typename FunctionTraits<decltype(MemFnT)>::ClassType>
113constexpr auto bind(ClassT* obj) noexcept
114{
115 using Traits = FunctionTraits<decltype(MemFnT)>;
116 using Slot = typename Traits::template Apply<BasicSlot>;
117 return Slot{}.template bind<MemFnT>(obj);
118}
119
120} // namespace util
121} // namespace toolbox
122
123#endif // TOOLBOX_UTIL_SLOT_HPP
void reset(std::nullptr_t=nullptr) noexcept
Definition Slot.hpp:85
constexpr auto & bind(ClassT *obj) noexcept
Definition Slot.hpp:67
constexpr BasicSlot & operator=(const BasicSlot &) noexcept=default
constexpr BasicSlot(std::nullptr_t=nullptr) noexcept
Definition Slot.hpp:39
constexpr BasicSlot(BasicSlot &&) noexcept=default
constexpr bool empty() const noexcept
Definition Slot.hpp:52
friend constexpr bool operator==(BasicSlot lhs, BasicSlot rhs) noexcept
Definition Slot.hpp:31
RetT operator()(ArgsT... args) const
Definition Slot.hpp:51
constexpr BasicSlot(const BasicSlot &) noexcept=default
constexpr auto & bind() noexcept
Definition Slot.hpp:57
constexpr auto & bind(ClassT *obj) noexcept
Definition Slot.hpp:77
friend constexpr bool operator!=(BasicSlot lhs, BasicSlot rhs) noexcept
Definition Slot.hpp:35
constexpr auto bind() noexcept
Definition Slot.hpp:97
Default case for functors and lambdas.
Definition Traits.hpp:27