14 constexpr T
abs(T value)
16 return value < 0 ? -value : value;
22 return abs(left - right) < eps;
28 return abs(value) < eps;
43 template<std::
integral T>
46 return (val & 1) == 1;
49 template<std::
integral T>
52 return (val & 1) == 0;
55 constexpr
int gcd(
int nFirst,
int nSecond)
57 if (nFirst == 0 || nSecond == 0)
62 const int nRemainder = nFirst % nSecond;
70 constexpr
int lcm(
int nFirst,
int nSecond)
72 if (nFirst == 0 || nSecond == 0)
76 nSecond =
abs(nSecond);
78 return nFirst /
gcd(nFirst, nSecond) * nSecond;
82 inline double pow(T number,
int nPower)
84 static_assert(std::is_integral_v<T> || std::is_floating_point_v<T>,
"Integral or floating point required");
86 const bool bNegativePower = nPower < 0;
87 const size_t nPositivePower =
static_cast<size_t>(std::abs(nPower));
90 switch (nPositivePower)
96 fResult =
static_cast<double>(number);
100 fResult =
static_cast<double>(number * number);
104 const std::bitset<std::numeric_limits<int>::digits> powerBitSet(nPositivePower);
106 std::array<double, std::numeric_limits<int>::digits> powers;
108 powers[0] =
static_cast<double>(number);
110 size_t nCurPower = 1;
111 size_t nCurIndex = 1;
113 while (nCurPower < nPositivePower)
115 powers[nCurIndex] = powers[nCurIndex - 1] * powers[nCurIndex - 1];
120 for (
size_t i = 0; i < nCurIndex; ++i)
121 if (powerBitSet.test(i))
122 fResult *= powers[i];
127 return bNegativePower ? 1.0 / fResult : fResult;
130 template<std::
integral I>
133 static_assert(std::is_integral_v<I>,
"Integral required");
138 std::bitset<std::numeric_limits<I>::digits> powers(
static_cast<size_t>(
abs(nValue)));
140 I nPow =
static_cast<I
>(std::numeric_limits<I>::digits - 1);
141 while (!powers.test(
static_cast<size_t>(nPow)))
150 QX_DISABLE_MSVC_WARNINGS(4018 4388);
152 template<
class T,
class compare_t>
153 constexpr
bool between(T left, T value, T right, compare_t compare)
155 if constexpr (std::is_enum_v<T>)
157 i64 l =
static_cast<i64
>(left);
158 i64 r =
static_cast<i64
>(right);
159 i64 v =
static_cast<i64
>(value);
160 return between(l, v, r, compare);
162 else if constexpr (std::is_integral_v<T> && std::is_same_v<compare_t, std::less_equal<>>)
164 return compare(
static_cast<size_t>(value - left), right - left);
166 else if constexpr (std::is_floating_point_v<T> && std::is_same_v<compare_t, std::less_equal<>>)
172 return compare(left, value) && compare(value, right);
176 template<
class T,
class compare_t>
177 constexpr
bool between(T left, T value, T right)
179 QX_PUSH_SUPPRESS_MSVC_WARNINGS(4388);
180 return between(left, value, right, compare_t());
181 QX_POP_SUPPRESS_WARNINGS();
184 QX_RESTORE_MSVC_WARNINGS(4018 4388);
constexpr int lcm(int nFirst, int nSecond)
Least common multiple.
constexpr bool between(T left, T value, T right, compare_t compare)
Checks if value is between left and right.
constexpr bool epsilon_greater_equal(T left, T right, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison function for a user defined epsilon values.
I maxpot(I nValue)
Max power of two in integer.
constexpr bool is_odd(T val)
Check if value is odd.
constexpr bool epsilon_zero(T value, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison with zero for a user defined epsilon values.
constexpr T abs(T value)
Constexpr absolute value.
constexpr bool epsilon_less_equal(T left, T right, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison function for a user defined epsilon values.
constexpr bool is_even(T val)
Check if value is even.
constexpr bool epsilon_equal(T left, T right, T eps=std::numeric_limits< T >::epsilon())
Constexpr comparison function for a user defined epsilon values.
double pow(T number, int nPower)
Power function for integer power.
constexpr int gcd(int nFirst, int nSecond)
Greatest common divisor.