1
0
mirror of https://github.com/VCMP-SqMod/SqMod.git synced 2024-11-08 16:57:16 +01:00
SqMod/source/Library/Numeric/Decimal.hpp
2016-06-05 03:54:15 +03:00

318 lines
11 KiB
C++

#ifndef _LIBRARY_NUMERIC_DECIMAL_HPP_
#define _LIBRARY_NUMERIC_DECIMAL_HPP_
// ------------------------------------------------------------------------------------------------
#include "Base/Utility.hpp"
// ------------------------------------------------------------------------------------------------
namespace SqMod {
/* ------------------------------------------------------------------------------------------------
* Simple decimal data type support. Mostly for working with databases that support this type.
*/
class Decimal
{
private:
// --------------------------------------------------------------------------------------------
Int64 m_Value; // Decimal value
Uint8 m_Precision; // Decimal precision
// --------------------------------------------------------------------------------------------
static const Int64 s_Factors[];
protected:
/* --------------------------------------------------------------------------------------------
* Validate the given precision.
*/
static void ValidatePrecision(Uint8 precision);
/* --------------------------------------------------------------------------------------------
* Convert a decimal to the same precision as this one.
*/
Int64 Convert(const Decimal & dec) const;
/* --------------------------------------------------------------------------------------------
* Validate against given decimal.
*/
Int32 Compare(const Decimal & o) const;
/* --------------------------------------------------------------------------------------------
* Result = (a * b) / d
*/
static Int64 MultiplyDivide(Int64 a, Int64 b, Int64 d)
{
if ((std::abs(a) <= std::numeric_limits< Int32 >::max()) ||
(std::abs(b) <= std::numeric_limits< Int32 >::max()))
{
return std::llround(static_cast< Float64 >(a * b) / static_cast< Float64 >(d));
}
return std::llround(static_cast< Float64 >(a) * static_cast< Float64 >(b) / static_cast< Float64 >(d));
}
public:
/* --------------------------------------------------------------------------------------------
* Default constructor.
*/
Decimal()
: m_Value(0), m_Precision(4)
{
/* ... */
}
/* --------------------------------------------------------------------------------------------
* String constructor.
*/
Decimal(CSStr value);
/* --------------------------------------------------------------------------------------------
* Numeric constructor.
*/
Decimal(SQInteger value, Uint8 precision);
#ifndef _SQ64
/* --------------------------------------------------------------------------------------------
* Numeric constructor.
*/
Decimal(Int64 value, Uint8 precision);
#endif // _SQ64
/* --------------------------------------------------------------------------------------------
* Copy constructor.
*/
Decimal(const Decimal & o) = default;
/* --------------------------------------------------------------------------------------------
* Move constructor.
*/
Decimal(Decimal && o) = default;
/* --------------------------------------------------------------------------------------------
* Destructor.
*/
~Decimal() = default;
/* --------------------------------------------------------------------------------------------
* Copy assignment operator.
*/
Decimal & operator = (const Decimal & o)
{
m_Value = Convert(o);
return *this;
}
/* --------------------------------------------------------------------------------------------
* Move assignment operator.
*/
Decimal & operator = (Decimal && o) = default;
/* --------------------------------------------------------------------------------------------
* Used by the script engine to compare two instances of this type.
*/
Int32 Cmp(const Decimal & o) const
{
return Compare(o);
}
/* --------------------------------------------------------------------------------------------
* Used by the script engine to convert an instance of this type to a string.
*/
CSStr ToString() const;
/* --------------------------------------------------------------------------------------------
* Used by the script engine to retrieve the name from instances of this type.
*/
static SQInteger Typename(HSQUIRRELVM vm);
/* --------------------------------------------------------------------------------------------
* Perform an equality comparison between to instances of this type.
*/
bool operator == (const Decimal & o) const
{
return m_Value == Convert(o);
}
/* --------------------------------------------------------------------------------------------
* Perform an inequality comparison between to instances of this type.
*/
bool operator != (const Decimal & o) const
{
return m_Value != Convert(o);
}
/* --------------------------------------------------------------------------------------------
* Perform a less than comparison between to instances of this type.
*/
bool operator < (const Decimal & o) const
{
return m_Value < Convert(o);
}
/* --------------------------------------------------------------------------------------------
* Perform a greater than comparison between to instances of this type.
*/
bool operator > (const Decimal & o) const
{
return m_Value > Convert(o);
}
/* --------------------------------------------------------------------------------------------
* Perform a less than or equal comparison between to instances of this type.
*/
bool operator <= (const Decimal & o) const
{
return m_Value <= Convert(o);
}
/* --------------------------------------------------------------------------------------------
* Perform a greater than or equal comparison between to instances of this type.
*/
bool operator >= (const Decimal & o) const
{
return m_Value >= Convert(o);
}
/* --------------------------------------------------------------------------------------------
* Perform an addition between to instances of this type.
*/
Decimal operator + (const Decimal & o) const
{
return Decimal(m_Value + Convert(o), m_Precision);
}
/* --------------------------------------------------------------------------------------------
* Perform an subtraction between to instances of this type.
*/
Decimal operator - (const Decimal & o) const
{
return Decimal(m_Value - Convert(o), m_Precision);
}
/* --------------------------------------------------------------------------------------------
* Perform an multiplication between to instances of this type.
*/
Decimal operator * (const Decimal & o) const
{
return Decimal(MultiplyDivide(m_Value, Convert(o), s_Factors[m_Precision]), m_Precision);
}
/* --------------------------------------------------------------------------------------------
* Perform an division between to instances of this type.
*/
Decimal operator / (const Decimal & o) const
{
return Decimal(MultiplyDivide(m_Value, s_Factors[m_Precision], Convert(o)), m_Precision);
}
/* --------------------------------------------------------------------------------------------
* Perform an assignment addition between to instances of this type.
*/
Decimal & operator += (const Decimal & o)
{
m_Value += Convert(o);
return *this;
}
/* --------------------------------------------------------------------------------------------
* Perform an subtraction addition between to instances of this type.
*/
Decimal & operator -= (const Decimal & o)
{
m_Value -= Convert(o);
return *this;
}
/* --------------------------------------------------------------------------------------------
* Perform an multiplication addition between to instances of this type.
*/
Decimal & operator *= (const Decimal & o)
{
m_Value -= MultiplyDivide(m_Value, Convert(o), s_Factors[m_Precision]);
return *this;
}
/* --------------------------------------------------------------------------------------------
* Perform an division addition between to instances of this type.
*/
Decimal & operator /= (const Decimal & o)
{
m_Value -= MultiplyDivide(m_Value, s_Factors[m_Precision], Convert(o));
return *this;
}
/* --------------------------------------------------------------------------------------------
* Convert a decimal to the same precision as this one.
*/
SLongInt GetConverted(const Decimal & dec) const;
/* --------------------------------------------------------------------------------------------
* Retrieve the factor.
*/
SLongInt GetFactor() const;
/* --------------------------------------------------------------------------------------------
* Returns integer value = real_value * (10 ^ precision)
*/
SLongInt GetUnbiased() const;
/* --------------------------------------------------------------------------------------------
* Retrieve the precision.
*/
Uint8 GetPrecision() const
{
return m_Precision;
}
/* --------------------------------------------------------------------------------------------
* Retrieve the absolute value.
*/
Decimal GetAbs() const
{
if (m_Value >= 0)
{
return *this;
}
// Calculate the absolute value
return Decimal(0, s_Factors[m_Precision]) - *this;
}
/* --------------------------------------------------------------------------------------------
* Retrieve the decimal value as a native script floating point value.
*/
SQFloat GetSqFloat() const
{
return static_cast< SQFloat >(m_Value) / static_cast< SQFloat >(m_Precision * s_Factors[m_Precision]);
}
/* --------------------------------------------------------------------------------------------
* Retrieve the decimal value as a single floating point value.
*/
Float32 GetFloat32() const
{
return static_cast< Float32 >(m_Value) / static_cast< Float32 >(m_Precision * s_Factors[m_Precision]);
}
/* --------------------------------------------------------------------------------------------
* Retrieve the decimal value as a double floating point value.
*/
Float64 GetFloat64() const
{
return static_cast< Float64 >(m_Value) / static_cast< Float64 >(m_Precision * s_Factors[m_Precision]);
}
/* --------------------------------------------------------------------------------------------
* Convert the decimal to a string and store it into the given buffer.
*/
void MakeString(CStr buffer, Uint32 size) const;
};
} // Namespace:: SqMod
#endif // _LIBRARY_NUMERIC_DECIMAL_HPP_