diff options
author | Stanisław Halik <sthalik@misaki.pl> | 2017-05-14 16:22:09 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2017-05-14 16:22:09 +0200 |
commit | 5c23666b58bb1dd4aea15c0d62a2f716d5be7f52 (patch) | |
tree | e6497e9b55c073be209ec673ef05e62bf57a2c8f /compat/functional.hpp | |
parent | 4701dd3b0c8323a11cf7d5ad09c579a9864a41bd (diff) | |
parent | c392181211b245e74292424500265323c960c1aa (diff) |
Merge branch 'unstable' into unstable
Diffstat (limited to 'compat/functional.hpp')
-rw-r--r-- | compat/functional.hpp | 86 |
1 files changed, 86 insertions, 0 deletions
diff --git a/compat/functional.hpp b/compat/functional.hpp new file mode 100644 index 00000000..893fe1a0 --- /dev/null +++ b/compat/functional.hpp @@ -0,0 +1,86 @@ +#pragma once + +#include <algorithm> +#include <iterator> +#include <type_traits> + +template<typename t> +using remove_qualifiers = std::remove_reference_t<std::remove_cv_t<t>>; + +namespace functools +{ + +template<typename seq_, typename = void> +struct reserver_ +{ + static inline void maybe_reserve_space(seq_&, unsigned) + { + //qDebug() << "nada"; + } +}; + +template<typename seq_> +struct reserver_<seq_, decltype(std::declval<seq_>().reserve(0u), (void)0)> +{ + static inline void maybe_reserve_space(seq_& seq, unsigned sz) + { + seq.reserve(sz); + } +}; + +template<typename seq_> +inline void maybe_reserve_space(seq_& seq, unsigned sz) +{ + reserver_<seq_, void>::maybe_reserve_space(seq, sz); +} + +} // ns + +template<typename t, t value_> +struct constant final +{ + using type = t; + constexpr type operator()() const noexcept + { + return value_; + } + static constexpr type value = value_; + + constant() = delete; +}; + +template<typename seq_, typename F> +auto map(F&& fun, const seq_& seq) +{ + using seq_type = remove_qualifiers<seq_>; + + seq_type ret; + std::back_insert_iterator<seq_type> it = std::back_inserter(ret); + + for (const auto& elt : seq) + it = fun(elt); + + return ret; +} + +template<typename seq_, typename F> +auto remove_if_not(F&& fun, const seq_& seq) +{ + using namespace functools; + + using seq_type = remove_qualifiers<seq_>; + using value_type = typename std::iterator_traits<decltype(std::begin(std::declval<seq_>()))>::value_type; + using fun_ret_type = decltype(fun(std::declval<const value_type&>())); + static_assert(std::is_convertible<fun_ret_type, bool>::value, "must return bool"); + + seq_type ret; + maybe_reserve_space(ret, seq.size()); + + std::back_insert_iterator<seq_type> it = std::back_inserter(ret); + + for (const value_type& elt : seq) + if (fun(elt)) + it = elt; + + return ret; +} |