diff options
| author | Stanislaw Halik <sthalik@misaki.pl> | 2017-05-04 16:37:00 +0200 | 
|---|---|---|
| committer | Stanislaw Halik <sthalik@misaki.pl> | 2017-05-10 11:19:22 +0200 | 
| commit | 07c87d1d45af4a3c8111445ab6e6a7863b235d3a (patch) | |
| tree | 2b0b05587e6efb2c77deed49b2116ad6a4925809 | |
| parent | de5e9b532abe95dae7b345bc816afafc819fe511 (diff) | |
compat: add map/filter template functions
| -rw-r--r-- | compat/functional.hpp | 75 | 
1 files changed, 75 insertions, 0 deletions
| diff --git a/compat/functional.hpp b/compat/functional.hpp new file mode 100644 index 00000000..e7229d6e --- /dev/null +++ b/compat/functional.hpp @@ -0,0 +1,75 @@ +#pragma once + +#include <algorithm> +#include <iterator> +#include <type_traits> + +namespace functools +{ + +constexpr void return_void(); + +template<typename t> +using remove_qualifiers = std::remove_reference_t<std::remove_cv_t<t>>; + +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), return_void())> +{ +    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 seq_, typename F> +auto map(F&& fun, const seq_& seq) +{ +    using seq_type = functools::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; +} | 
