From 38ab48b1882f36ed7eb7e50c4fb46ce5d376fbc3 Mon Sep 17 00:00:00 2001 From: Steven Le Rouzic Date: Wed, 26 Feb 2025 20:01:45 +0100 Subject: Fix some deducing-this & functional stuff, add invocable concept --- asl/base/defer.hpp | 6 ++---- asl/base/functional.hpp | 19 ++++++++++++++++--- asl/base/functional_tests.cpp | 24 +++++++++++++++++++----- asl/types/option.hpp | 25 ++++++++++--------------- 4 files changed, 47 insertions(+), 27 deletions(-) (limited to 'asl') diff --git a/asl/base/defer.hpp b/asl/base/defer.hpp index b6d52af..e881d8a 100644 --- a/asl/base/defer.hpp +++ b/asl/base/defer.hpp @@ -6,8 +6,7 @@ namespace asl { -// @Todo Add invokable check -template +template class DeferCallback { Callback m_callback; @@ -36,8 +35,7 @@ public: struct DeferFactory { - // @Todo Add invokable check - template + template DeferCallback operator<<(Callback&& callback) const { return DeferCallback(ASL_FWD(callback)); diff --git a/asl/base/functional.hpp b/asl/base/functional.hpp index d820bce..b4b5312 100644 --- a/asl/base/functional.hpp +++ b/asl/base/functional.hpp @@ -7,6 +7,7 @@ namespace asl { template constexpr auto invoke(is_func auto C::* f, auto&& self, Args&&... args) + -> decltype((self.*f)(ASL_FWD(args)...)) requires requires { (self.*f)(ASL_FWD(args)...); } @@ -16,6 +17,7 @@ constexpr auto invoke(is_func auto C::* f, auto&& self, Args&&... args) template constexpr auto invoke(is_func auto C::* f, auto* self, Args&&... args) + -> decltype((self->*f)(ASL_FWD(args)...)) requires requires { (self->*f)(ASL_FWD(args)...); } @@ -25,6 +27,7 @@ constexpr auto invoke(is_func auto C::* f, auto* self, Args&&... args) template constexpr auto invoke(is_object auto C::* m, auto&& self, Args&&...) + -> decltype(self.*m) requires ( sizeof...(Args) == 0 && requires { self.*m; } @@ -35,6 +38,7 @@ constexpr auto invoke(is_object auto C::* m, auto&& self, Args&&...) template constexpr auto invoke(is_object auto C::* m, auto* self, Args&&...) + -> decltype(self->*m) requires ( sizeof...(Args) == 0 && requires { self->*m; } @@ -45,6 +49,7 @@ constexpr auto invoke(is_object auto C::* m, auto* self, Args&&...) template constexpr auto invoke(auto&& f, Args&&... args) + -> decltype(f(ASL_FWD(args)...)) requires requires { f(ASL_FWD(args)...); } @@ -52,14 +57,22 @@ constexpr auto invoke(auto&& f, Args&&... args) return ASL_FWD(f)(ASL_FWD(args)...); } -template struct _result_of_helper; +template +struct _invoke_result_helper; template -struct _result_of_helper +struct _invoke_result_helper { using type = decltype(invoke(declval(), declval()...)); }; -template using result_of_t = _result_of_helper::type; +template +using invoke_result_t = _invoke_result_helper::type; + +template +concept invocable = requires (F&& f, Args&&... args) +{ + invoke(ASL_FWD(f), ASL_FWD(args)...); +}; } // namespace asl diff --git a/asl/base/functional_tests.cpp b/asl/base/functional_tests.cpp index 92c5c7b..991c3c1 100644 --- a/asl/base/functional_tests.cpp +++ b/asl/base/functional_tests.cpp @@ -24,11 +24,25 @@ static int some_func1(int x) { return x + 1; } [[maybe_unused]] static float some_func1(float x) { return x + 1; } static int some_func2(int x, int b) { return x + b; } -static_assert(asl::same_as, int64_t>); -static_assert(asl::same_as, int>); -static_assert(asl::same_as(some_func1))(float)>, float>); -static_assert(asl::same_as, void>); -static_assert(asl::same_as, int>); +static_assert(asl::same_as, int>); +static_assert(asl::same_as, int>); +static_assert(asl::same_as, int64_t>); +static_assert(asl::same_as, int>); +static_assert(asl::same_as(some_func1)), float>, float>); +static_assert(asl::same_as, void>); +static_assert(asl::same_as, const int&>); + +static_assert(asl::invocable); +static_assert(!asl::invocable); +static_assert(asl::invocable); +static_assert(!asl::invocable); +static_assert(asl::invocable); +static_assert(asl::invocable); +static_assert(!asl::invocable); +static_assert(asl::invocable(some_func1)), float>); +static_assert(asl::invocable); +static_assert(!asl::invocable); +static_assert(asl::invocable); ASL_TEST(invoke_member_function) { diff --git a/asl/types/option.hpp b/asl/types/option.hpp index e79c09c..b32483b 100644 --- a/asl/types/option.hpp +++ b/asl/types/option.hpp @@ -395,32 +395,27 @@ public: return value(); } - template< - typename F, - typename Self, - typename Result = result_of_t)> - > + template constexpr auto and_then(this Self&& self, F&& f) - requires is_option { + using Result = invoke_result_t>; + static_assert(is_option); + if (self.has_value()) { - return invoke(ASL_FWD(f), ASL_FWD_LIKE(decltype(self), ASL_FWD(self).value())); + return invoke(ASL_FWD(f), ASL_FWD(self).value()); } return Result{ asl::nullopt }; } - template< - typename F, - typename Self, - typename Result = result_of_t)> - > + template constexpr auto transform(this Self&& self, F&& f) { + using Result = invoke_result_t>; if (self.has_value()) { return option>{ - invoke(ASL_FWD(f), ASL_FWD_LIKE(decltype(self), ASL_FWD(self).value())) + invoke(ASL_FWD(f), ASL_FWD(self).value()) }; } return option>{ asl::nullopt }; @@ -428,14 +423,14 @@ public: template constexpr option or_else(F&& f) const& - requires same_as>, option> + requires same_as>, option> { return has_value() ? *this : invoke(ASL_FWD(f)); } template constexpr option or_else(F&& f) && - requires same_as>, option> + requires same_as>, option> { return has_value() ? ASL_MOVE(*this) : invoke(ASL_FWD(f)); } -- cgit