/*
Copyright 2014 Michele "King_DuckZ" Santullo
This file is part of CloonelJump.
CloonelJump 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.
CloonelJump 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 CloonelJump. If not, see .
*/
#ifndef idid0528646832E04CF08E9785B66CFE0BD1
#define idid0528646832E04CF08E9785B66CFE0BD1
#include
#include
#include
#include
#include
#if !defined(NDEBUG)
#include
#endif
namespace cloonel {
template
class Vector {
template friend class Vector;
public:
Vector ( void ) noexcept(noexcept(T())) = default;
explicit Vector ( T parValue ) noexcept(noexcept(T()) && noexcept(parValue=parValue));
template explicit Vector ( const Vector& parOther ) noexcept(noexcept(T()) && noexcept(const_cast(parOther.m_mem[0])=T()));
template > Vector ( T parX, T parY ) noexcept : m_mem {parX, parY} {}
template > Vector ( T parX, T parY, T parZ ) noexcept : m_mem {parX, parY, parZ} {}
template > Vector ( T parX, T parY, T parZ, T parW ) noexcept : m_mem {parX, parY, parZ, parW} {}
~Vector ( void ) noexcept = default;
enum {
Dimension = S
};
template = S> > T& x ( void ) { return m_mem[0]; }
template = S> > const T& x ( void ) const { return m_mem[0]; }
template = 2 and 4 >= S> > T& y ( void ) { return m_mem[1]; }
template = 2 and 4 >= S> > const T& y ( void ) const { return m_mem[1]; }
template = 3 and 4 >= S> > T& z ( void ) { return m_mem[2]; }
template = 3 and 4 >= S> > const T& z ( void ) const { return m_mem[2]; }
template = 4 and 4 >= S> > T& w ( void ) { return m_mem[3]; }
template = 4 and 4 >= S> > const T& w ( void ) const { return m_mem[3]; }
template const Vector& operator+= ( const Vector& parOther );
template const Vector& operator-= ( const Vector& parOther );
template const Vector& operator*= ( const Vector& parOther );
template const Vector& operator/= ( const Vector& parOther );
template const Vector& operator+= ( U parOther );
template const Vector& operator-= ( U parOther );
template const Vector& operator*= ( U parOther );
template const Vector& operator/= ( U parOther );
T& operator[] ( uint32_t parIndex ) { assert(parIndex < S); return m_mem[parIndex]; }
const T& operator[] ( uint32_t parIndex ) const { assert(parIndex < S); return m_mem[parIndex]; }
private:
T m_mem[S];
};
template
Vector::type, S> operator+ ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
Vector::type, S> operator- ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
Vector::type, S> operator* ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
Vector::type, S> operator/ ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator+ ( U parA, const Vector& parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator- ( U parA, const Vector& parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator* ( U parA, const Vector& parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator/ ( U parA, const Vector& parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator+ ( const Vector& parA, U parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator- ( const Vector& parA, U parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator* ( const Vector& parA, U parB ) __attribute__((pure));
template ::value>::type>
Vector::type, S> operator/ ( const Vector& parA, U parB ) __attribute__((pure));
template
bool operator< ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
bool operator> ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
bool operator<= ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
bool operator>= ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
bool operator== ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
bool operator!= ( const Vector& parA, const Vector& parB ) __attribute__((pure));
template
Vector operator- ( Vector parOperand ) __attribute__((pure));
typedef Vector float2;
typedef Vector ushort2;
typedef Vector int2;
#if !defined(NDEBUG)
template
std::ostream& operator<< ( std::ostream& parStream, const Vector& parVector ) {
parStream << "<";
for (uint32_t z = 0; z < S - 1; ++z) {
parStream << parVector[z] << ",";
}
parStream << parVector[S - 1] << ">";
return parStream;
}
#endif
namespace implem {
template
struct CategorizeTypes {
typedef typename std::common_type::type CommonType;
typedef typename std::conditional::value, U, T>::type OtherType;
};
template ::value>
struct DoOperation {
static Vector do_mul ( Vector parLho, const Vector& parRho ) {
parLho *= parRho;
return parLho;
}
static Vector do_div ( Vector parLho, const Vector& parRho ) {
parLho /= parRho;
return parLho;
}
static Vector do_sum ( Vector parLho, const Vector& parRho ) {
parLho += parRho;
return parLho;
}
static Vector do_sub ( Vector parLho, const Vector& parRho ) {
parLho -= parRho;
return parLho;
}
};
template
struct DoOperation {
static Vector do_mul ( Vector parLho, const Vector& parRho ) {
parLho *= parRho;
return parLho;
}
static Vector do_mul ( const Vector& parLho, const Vector& parRho ) {
Vector ret(parLho);
ret *= parRho;
return ret;
}
static Vector do_div ( Vector parLho, const Vector& parRho ) {
parLho /= parRho;
return parLho;
}
static Vector do_div ( const Vector& parLho, const Vector& parRho ) {
Vector ret(parLho);
ret /= parRho;
return ret;
}
static Vector do_sum ( Vector parLho, const Vector& parRho ) {
parLho += parRho;
return parLho;
}
static Vector do_sum ( const Vector& parLho, const Vector& parRho ) {
Vector ret(parLho);
ret += parRho;
return ret;
}
static Vector do_sub ( Vector parLho, const Vector& parRho ) {
parLho -= parRho;
return parLho;
}
static Vector do_sub ( const Vector& parLho, const Vector& parRho ) {
Vector ret(parLho);
ret -= parRho;
return ret;
}
};
} //namespace implem
} //namespace cloonel
#include "vector.inl"
#endif