/******************************************************************************* * * TRIQS: a Toolbox for Research in Interacting Quantum Systems * * Copyright (C) 2013 by O. Parcollet * * TRIQS is free software: you can redistribute it and/or modify it under the * terms of the GNU General Public License as published by the Free Software * Foundation, either version 3 of the License, or (at your option) any later * version. * * TRIQS is distributed in the hope that it will be useful, but WITHOUT ANY * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License along with * TRIQS. If not, see . * ******************************************************************************/ #ifndef TRIQS_UTILITY_TUPLE_TOOLS_H #define TRIQS_UTILITY_TUPLE_TOOLS_H #include #include #include namespace triqs { namespace tuple { /** * apply(f, t) * f : a callable object * t a tuple * Returns : f(t[0], t[1], ...) * Equivalent to f(*t) in python .... * Q : what about constructor */ template struct apply_impl { template auto operator()(F && f, T const & t, Args && ... args) DECL_AND_RETURN( apply_impl()(std::forward(f),t, std::get(t), std::forward(args)...)); }; template struct apply_impl<-1,T> { template auto operator()(F && f, T const & t, Args && ... args) DECL_AND_RETURN( std::forward(f)(std::forward(args)...)); }; template auto apply (F && f, T const & t) DECL_AND_RETURN( apply_impl::value-1,T>()(std::forward(f),t)); //template //ReturnType apply(ReturnType(ClassType::*f)(Args...) const, T const & t) { return apply([f](Args const & ... args) { return (*f)(args...);} ,t);} template ReturnType apply( ReturnType(*f)(Args...), T const & t) { return apply([f](Args const & ... args) { return (*f)(args...);} ,t);} /** * for_each(f, t) * f: a callable object * t: a tuple * calls f on all tuple elements: f(x) for all x in t */ template struct for_each_impl { template void operator()(T const & t, F && f) { f(std::get::value-1-pos>(t)); for_each_impl()(t, f); } }; template<> struct for_each_impl<0> { template void operator() (T const & t, F && f) { f(std::get::value-1>(t)); } }; template void for_each(T const & t, F && f) { for_each_impl::value-1>()(t, f); } /* for_each_enumerate(f, t) * f: a callable object * t: a tuple * calls f on all tuple elements: f(x,n) for all x in t */ template struct for_each_enumerate_impl { template void operator()(T const & t, F && f) { f(std::get::value-1-pos>(t),std::tuple_size::value-1-pos); for_each_impl()(t, f); } }; template<> struct for_each_enumerate_impl<0> { template void operator() (T const & t, F && f) { f(std::get::value-1>(t), std::tuple_size::value-1); } }; template void for_each_enumerate(T const & t, F && f) { for_each_enumerate_impl::value-1>()(t, f); } /** * apply_on_zip(f, t1,t2) * f : a callable object * t1, t2 two tuples of the same size * Returns : [f(i,j) for i,j in zip(t1,t2)] */ template struct apply_on_zip_impl { template auto operator()(F && f, T1 const & t1, T2 const & t2, Args && ... args) DECL_AND_RETURN( apply_on_zip_impl()(std::forward(f),t1, t2, f(std::get(t1),std::get(t2)), std::forward(args)...)); }; template struct apply_on_zip_impl<-1,T1,T2> { template auto operator()(F && f, T1 const & t1, T2 const & t2, Args && ... args) DECL_AND_RETURN( std::make_tuple(std::forward(args)...)); }; template auto apply_on_zip (F && f,T1 const & t1, T2 const & t2) DECL_AND_RETURN( apply_on_zip_impl::value-1,T1,T2>()(std::forward(f),t1,t2)); /** * apply_on_tuple(f, t1,t2,t3) * f : a callable object * t1, t2 two tuples of the same size * Returns : [f(i,j) for i,j in zip(t1,t2)] */ template struct apply_on_zip3_impl { template auto operator()(F && f, T1 const & t1, T2 const & t2, T3 const & t3, Args && ... args) DECL_AND_RETURN( apply_on_zip3_impl()(std::forward(f),t1, t2, t3, f(std::get(t1),std::get(t2),std::get(t3)), std::forward(args)...)); }; template struct apply_on_zip3_impl<-1,T1,T2,T3> { template auto operator()(F && f, T1 const & t1, T2 const & t2, T3 const & t3, Args && ... args) DECL_AND_RETURN( std::make_tuple(std::forward(args)...)); }; template auto apply_on_zip3 (F && f,T1 const & t1, T2 const & t2, T3 const & t3) DECL_AND_RETURN( apply_on_zip3_impl::value-1,T1,T2,T3>()(std::forward(f),t1,t2,t3)); /** * fold(f, t1, init) * f : a callable object * t a tuple * Returns : f(x0,f(x1,f(....)) on the tuple */ template struct fold_impl { template auto operator()(F && f, T & t, R && r ) DECL_AND_RETURN( fold_impl()(std::forward(f),t, f(std::get(t), std::forward(r)))); }; template struct fold_impl { template R operator()(F && f, T & t, R && r) {return std::forward(r);} }; template auto fold (F && f,T & t, R && r) DECL_AND_RETURN( fold_impl::value,std::tuple_size::value-1,T>()(std::forward(f),t,std::forward(r))); template auto fold (F && f,T const & t, R && r) DECL_AND_RETURN( fold_impl::value,std::tuple_size::value-1,T const>()(std::forward(f),t,std::forward(r))); /** * fold_on_zip(f, t1, t2, init) * f : a callable object * t1, t2 two tuples of the same size * Returns : f(x0,y0,f(x1,y1,,f(....)) for t1 = (x0,x1 ...) and t2 = (y0,y1...). */ template struct fold_on_zip_impl { template auto operator()(F && f, T1 const & t1, T2 const & t2, R && r ) DECL_AND_RETURN( fold_on_zip_impl()(std::forward(f),t1,t2, f(std::get(t1), std::get(t2), std::forward(r)))); }; template struct fold_on_zip_impl<-1,T1,T2> { template R operator()(F && f, T1 const & t1, T2 const & t2, R && r) {return std::forward(r);} }; template auto fold_on_zip (F && f,T1 const & t1, T2 const & t2, R && r) DECL_AND_RETURN( fold_on_zip_impl::value-1,T1,T2>()(std::forward(f),t1,t2,std::forward(r))); /* * print a tuple */ template struct __s {}; template void print_tuple_impl (std::ostream& os, T const& t, std::integral_constant ) {} template void print_tuple_impl (std::ostream& os, T const& t, std::integral_constant ) { os << std::get(t); if (rpos>0) os << ','; print_tuple_impl(os, t, std::integral_constant()); } }} namespace std { template std::ostream & operator << (std::ostream & os, std::tuple const & t) { os << "("; constexpr int L = sizeof...(T); triqs::tuple::print_tuple_impl(os,t,std::integral_constant()); return os << ")"; } } #endif