count.hpp
Go to the documentation of this file.
1 //
2 // Copyright (C) 2014, 2015 Ableton AG, Berlin. All rights reserved.
3 //
4 // Permission is hereby granted, free of charge, to any person obtaining a
5 // copy of this software and associated documentation files (the "Software"),
6 // to deal in the Software without restriction, including without limitation
7 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 // and/or sell copies of the Software, and to permit persons to whom the
9 // Software is furnished to do so, subject to the following conditions:
10 //
11 // The above copyright notice and this permission notice shall be included in
12 // all copies or substantial portions of the Software.
13 //
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
19 // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
20 // DEALINGS IN THE SOFTWARE.
21 //
22 
27 #pragma once
28 
32 
33 namespace atria {
34 namespace xform {
35 
36 namespace detail {
37 
38 struct count_rf_gen
39 {
40  template <typename ReducingFnT,
41  typename InitialT,
42  typename StepT>
43  struct apply
44  {
45  ReducingFnT step;
46  InitialT initial;
47  StepT count_step;
48 
49  template <typename StateT, typename ...InputTs>
50  auto operator() (StateT&& s, InputTs&& ...is)
51  -> decltype(wrap_state(step(state_unwrap(s), is..., initial), initial))
52  {
53  auto count = state_data(std::forward<StateT>(s), constantly(initial));
54  auto next = count;
55  next += count_step;
56  return wrap_state(
57  step(state_unwrap(std::forward<StateT>(s)),
58  std::forward<InputTs>(is)...,
59  std::move(count)),
60  std::move(next));
61  }
62  };
63 };
64 
65 } // namespace detail
66 
67 template <typename T1, typename T2>
68 using count_t = transducer_impl<detail::count_rf_gen, T1, T2>;
69 
77 template <typename InitT=std::size_t, typename StepT=InitT>
78 constexpr auto count(InitT init = InitT{0}, StepT step = StepT{1})
79  -> count_t<InitT, StepT>
80 {
81  return count_t<InitT, StepT> { std::move(init), std::move(step) };
82 }
83 
84 } // namespace xform
85 } // namespace atria
constexpr auto count(InitT init=InitT{0}, StepT step=StepT{1}) -> count_t< InitT, StepT >
Generator transducer produces a sequence:
Definition: count.hpp:78
auto state_unwrap(T &&s) -> decltype(state_traits_t< T >::unwrap(std::forward< T >(s)))
Convenience function for calling state_traits::unwrap
auto constantly(T &&value) -> constantly_t< estd::decay_t< T > >
Similar to clojure.core/constantly.
Definition: constantly.hpp:56
auto state_data(T &&s, D &&d) -> decltype(state_traits_t< T >::data(std::forward< T >(s), std::forward< D >(d)))
Convenience function for calling state_traits::data
C++ amazing templates and reusable implementations awesomeness.
Definition: _doc.hpp:35
auto wrap_state(StateT &&next, DataT &&data=DataT{}) -> state_wrapper< TagT, estd::decay_t< StateT >, estd::decay_t< DataT > >
Given a tag TagT and a state next and associated data, returns a state_wrapper instance.
Fork me on GitHub