coord/ops: ensure coord funcs only receive coords

This commit is contained in:
Danny Robson 2016-03-14 19:54:25 +11:00
parent 0cfcd7b439
commit 12b526dff7

View File

@ -57,6 +57,10 @@ namespace util {
template <> struct is_coord<vector> : std::true_type { }; template <> struct is_coord<vector> : std::true_type { };
template <> struct is_coord<colour> : std::true_type { }; template <> struct is_coord<colour> : std::true_type { };
template <template <size_t,typename> class K>
constexpr bool
is_coord_v = is_coord<K>::value;
/////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////
// vector operators // vector operators
#define ELEMENT_OP(OP) \ #define ELEMENT_OP(OP) \
@ -66,17 +70,17 @@ namespace util {
typename U, \ typename U, \
template <size_t,typename> class A, \ template <size_t,typename> class A, \
template <size_t,typename> class B, \ template <size_t,typename> class B, \
typename = typename std::enable_if< \ typename = std::enable_if_t< \
is_coord<A>::value && is_coord<B>::value, \ is_coord_v<A> && is_coord_v<B>, \
void \ void \
>::type \ > \
> \ > \
constexpr \ constexpr \
auto \ auto \
operator OP (A<S,T> a, B<S,U> b) \ operator OP (A<S,T> a, B<S,U> b) \
{ \ { \
typename coord::traits<A,B>::template result< \ typename coord::traits<A,B>::template result< \
S,typename std::common_type<T,U>::type \ S,std::common_type_t<T,U> \
> out {}; \ > out {}; \
for (size_t i = 0; i < S; ++i) \ for (size_t i = 0; i < S; ++i) \
out[i] = a[i] OP b[i]; \ out[i] = a[i] OP b[i]; \
@ -89,15 +93,14 @@ namespace util {
typename U, \ typename U, \
template <size_t,typename> class A, \ template <size_t,typename> class A, \
template <size_t,typename> class B, \ template <size_t,typename> class B, \
typename = typename std::enable_if< \ typename = std::enable_if_t< \
is_coord<A>::value && \ is_coord_v<A> && \
is_coord<B>::value && \ is_coord_v<B> && \
std::is_same< \ std::is_same< \
typename std::common_type<T,U>::type, \ std::common_type_t<T,U>, T \
T \
>::value, \ >::value, \
void \ void \
>::type \ > \
> \ > \
auto& \ auto& \
operator PASTE(OP,=) (A<S,T>& a, B<S,U> b) \ operator PASTE(OP,=) (A<S,T>& a, B<S,U> b) \
@ -122,15 +125,15 @@ namespace util {
typename T, \ typename T, \
typename U, \ typename U, \
template <size_t,typename> class K, \ template <size_t,typename> class K, \
typename = typename std::enable_if< \ typename = std::enable_if_t< \
is_coord<K>::value,void \ is_coord_v<K>,void \
>::type \ > \
> \ > \
constexpr \ constexpr \
auto \ auto \
operator OP (U u, K<S,T> k) \ operator OP (U u, K<S,T> k) \
{ \ { \
K<S,typename std::common_type<T,U>::type> out{};\ K<S,std::common_type_t<T,U>> out{}; \
\ \
for (size_t i = 0; i < S; ++i) \ for (size_t i = 0; i < S; ++i) \
out[i] = u OP k[i]; \ out[i] = u OP k[i]; \
@ -142,15 +145,15 @@ namespace util {
typename T, \ typename T, \
typename U, \ typename U, \
template <size_t,typename> class K, \ template <size_t,typename> class K, \
typename = typename std::enable_if< \ typename = std::enable_if_t< \
is_coord<K>::value,void \ is_coord<K>::value,void \
>::type \ > \
> \ > \
constexpr \ constexpr \
auto \ auto \
operator OP (K<S,T> k, U u) \ operator OP (K<S,T> k, U u) \
{ \ { \
K<S,typename std::common_type<T,U>::type> out{};\ K<S,std::common_type_t<T,U>> out {}; \
\ \
for (size_t i = 0; i < S; ++i) \ for (size_t i = 0; i < S; ++i) \
out[i] = k[i] OP u; \ out[i] = k[i] OP u; \
@ -177,17 +180,18 @@ namespace util {
typename T, \ typename T, \
typename U, \ typename U, \
template <size_t,typename> class K, \ template <size_t,typename> class K, \
typename = typename std::enable_if< \ typename = std::enable_if_t< \
is_coord<K>::value, \ is_coord<K>::value, \
void \ void \
>::type \ > \
> \ > \
typename std::enable_if< \ std::enable_if_t< \
std::is_same< \ std::is_same< \
T, \ T, \
typename std::common_type<T,U>::type>::value, \ std::common_type_t<T,U> \
>::value, \
K<S,T> \ K<S,T> \
>::type& \ >& \
operator OP (K<S,T> &k, U u) \ operator OP (K<S,T> &k, U u) \
{ \ { \
for (size_t i = 0; i < S; ++i) \ for (size_t i = 0; i < S; ++i) \
@ -211,7 +215,10 @@ namespace util {
template < \ template < \
size_t S, \ size_t S, \
typename T, \ typename T, \
template <size_t,typename> class K \ template <size_t,typename> class K, \
typename = std::enable_if_t< \
is_coord_v<K>, void \
> \
> \ > \
constexpr \ constexpr \
auto \ auto \
@ -240,17 +247,20 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
bool bool
operator== (K<S,T> a, K<S,T> b) operator== (const K<S,T> a, const K<S,T> b)
{ {
bool (*predicate)(const T&, const T&) = almost_equal; bool (*predicate)(const T&, const T&) = almost_equal;
return std::equal (std::begin (a), return std::equal (std::cbegin (a),
std::end (a), std::cend (a),
std::begin (b), std::cbegin (b),
predicate); predicate);
} }
@ -259,7 +269,10 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
bool bool
@ -279,10 +292,10 @@ namespace util {
typename U typename U
> >
constexpr constexpr
vector<S,typename std::common_type<T,U>::type> vector<S,std::common_type_t<T,U>>
operator- (point<S,T> a, point<S,U> b) operator- (point<S,T> a, point<S,U> b)
{ {
vector<S,typename std::common_type<T,U>::type> out {}; vector<S,std::common_type_t<T,U>> out {};
for (size_t i = 0; i < S; ++i) for (size_t i = 0; i < S; ++i)
out[i] = a[i] - b[i]; out[i] = a[i] - b[i];
return out; return out;
@ -296,7 +309,7 @@ namespace util {
typename U typename U
> >
constexpr constexpr
vector<S,typename std::common_type<T,U>::type> vector<S,std::common_type_t<T,U>>
operator- (U u, point<S,T> p) operator- (U u, point<S,T> p)
{ {
return point<S,U> {u} - p; return point<S,U> {u} - p;
@ -322,13 +335,13 @@ namespace util {
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class A, template <size_t,typename> class A,
template <size_t,typename> class B template <size_t,typename> class B,
typename = std::enable_if_t<
is_coord_v<A> && is_coord_v<B>, void
>
> >
constexpr constexpr
typename std::enable_if< T
is_coord<A>::value && is_coord<B>::value,
T
>::type
dot (A<S,T> a, B<S,T> b) dot (A<S,T> a, B<S,T> b)
{ {
return dot<S,T> (a.data, b.data); return dot<S,T> (a.data, b.data);
@ -338,10 +351,13 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
typename std::enable_if<is_coord<K>::value,T>::type T
dot (K<S,T> a, const T (&b)[S]) dot (K<S,T> a, const T (&b)[S])
{ {
return dot<S,T> (a.data, b); return dot<S,T> (a.data, b);
@ -351,13 +367,13 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
typename std::enable_if< T
is_coord<K>::value,
T
>::type
dot (const T (&a)[S], K<S,T> b) dot (const T (&a)[S], K<S,T> b)
{ {
return dot<S,T> (a, b.data); return dot<S,T> (a, b.data);
@ -368,7 +384,10 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
K<S,T> K<S,T>
@ -383,7 +402,10 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
K<S,T> K<S,T>
@ -402,7 +424,10 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
K<S,T> K<S,T>
@ -420,7 +445,10 @@ namespace util {
template < template <
size_t S, size_t S,
typename T, typename T,
template <size_t,typename> class K template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
> >
constexpr constexpr
K<S,T> K<S,T>
@ -434,18 +462,36 @@ namespace util {
///------------------------------------------------------------------------ ///------------------------------------------------------------------------
template <size_t S, typename T, template<size_t,typename> class K> template <
size_t S,
typename T,
template<size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
>
constexpr constexpr
T T
min (K<S,T> k) min (const K<S,T> k)
{ return *std::min_element (k.begin (), k.end ()); } {
return *std::min_element (std::cbegin (k), std::cend (k));
}
template <size_t S, typename T, template<size_t,typename> class K> template <
size_t S,
typename T,
template<size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
>
constexpr constexpr
T T
max (K<S,T> k) max (const K<S,T> k)
{ return *std::max_element (k.begin (), k.end ()); } {
return *std::max_element (std::cbegin (k), std::cend (k));
}
//------------------------------------------------------------------------- //-------------------------------------------------------------------------
@ -455,7 +501,10 @@ namespace util {
typename T, \ typename T, \
typename U, \ typename U, \
template <size_t,typename> class A, \ template <size_t,typename> class A, \
template <size_t,typename> class B \ template <size_t,typename> class B, \
typename = std::enable_if_t< \
is_coord_v<A> && is_coord_v<B>, void \
> \
> \ > \
constexpr \ constexpr \
vector<S,bool> \ vector<S,bool> \
@ -480,7 +529,10 @@ namespace util {
size_t S, \ size_t S, \
typename T, \ typename T, \
typename U, \ typename U, \
template <size_t,typename> class K \ template <size_t,typename> class K, \
typename = std::enable_if_t< \
is_coord_v<K>, void \
> \
> \ > \
constexpr \ constexpr \
vector<S,bool> \ vector<S,bool> \
@ -501,31 +553,51 @@ namespace util {
//------------------------------------------------------------------------- //-------------------------------------------------------------------------
template <size_t S, template <size_t,typename> class K> template <
size_t S,
template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
>
constexpr constexpr
bool bool
any (const K<S,bool> k) any (const K<S,bool> k)
{ {
return std::any_of (std::cbegin (k), std::cbegin (k), identity<bool>); return std::any_of (std::cbegin (k),
std::cbegin (k),
identity<bool>);
} }
//------------------------------------------------------------------------- //-------------------------------------------------------------------------
template <size_t S, template <size_t,typename> class K> template <
size_t S,
template <size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K>, void
>
>
constexpr constexpr
bool bool
all (const K<S,bool> k) all (const K<S,bool> k)
{ {
return std::all_of (std::cbegin (k), std::cbegin (k), identity<bool>); return std::all_of (std::cbegin (k),
std::cbegin (k),
identity<bool>);
} }
/////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////
template <size_t S, typename T, template<size_t,typename> class K> template <
size_t S,
typename T,
template<size_t,typename> class K,
typename = std::enable_if_t<
is_coord_v<K> && std::is_floating_point<T>::value, void
>
>
constexpr constexpr
typename std::enable_if< K<S,T>
std::is_floating_point<T>::value,
K<S,T>
>::type
floor (const K<S,T> k) floor (const K<S,T> k)
{ {
T (*floor_func)(T) = std::floor; T (*floor_func)(T) = std::floor;