using System;
namespace NUnit.Framework.Constraints
{
	/// 
	/// The Numerics class contains common operations on numeric values.
	/// 
	public class Numerics
	{
		#region Numeric Type Recognition
		/// 
		/// Checks the type of the object, returning true if
		/// the object is a numeric type.
		/// 
		/// The object to check
		/// true if the object is a numeric type
		public static bool IsNumericType(Object obj)
		{
			return IsFloatingPointNumeric( obj ) || IsFixedPointNumeric( obj );
		}
		/// 
		/// Checks the type of the object, returning true if
		/// the object is a floating point numeric type.
		/// 
		/// The object to check
		/// true if the object is a floating point numeric type
		public static bool IsFloatingPointNumeric(Object obj)
		{
			if (null != obj)
			{
				if (obj is double) return true;
				if (obj is float) return true;
				if (obj is System.Double) return true;
				if (obj is System.Single) return true;
			}
			return false;
		}
		/// 
		/// Checks the type of the object, returning true if
		/// the object is a fixed point numeric type.
		/// 
		/// The object to check
		/// true if the object is a fixed point numeric type
		public static bool IsFixedPointNumeric(Object obj)
		{
			if (null != obj)
			{
				if (obj is byte) return true;
				if (obj is sbyte) return true;
				if (obj is decimal) return true;
				if (obj is int) return true;
				if (obj is uint) return true;
				if (obj is long) return true;
				if (obj is short) return true;
				if (obj is ushort) return true;
				if (obj is System.Byte) return true;
				if (obj is System.SByte) return true;
				if (obj is System.Decimal) return true;
				if (obj is System.Int32) return true;
				if (obj is System.UInt32) return true;
				if (obj is System.Int64) return true;
				if (obj is System.UInt64) return true;
				if (obj is System.Int16) return true;
				if (obj is System.UInt16) return true;
			}
			return false;
		}
		#endregion
		#region Numeric Equality
        /// 
        /// Test two numeric values for equality, performing the usual numeric 
        /// conversions and using a provided or default tolerance. If the value 
        /// referred to by tolerance is null, this method may set it to a default.
        /// 
        /// The expected value
        /// The actual value
        /// A reference to the numeric tolerance in effect
        /// True if the values are equal
		public static bool AreEqual( object expected, object actual, ref object tolerance )
		{
            if (IsFloatingPointNumeric(expected) || IsFloatingPointNumeric(actual))
                return AreEqual(Convert.ToDouble(expected), Convert.ToDouble(actual), ref tolerance);
			if ( expected is decimal || actual is decimal )
				return AreEqual( Convert.ToDecimal(expected), Convert.ToDecimal(actual), Convert.ToDecimal(tolerance) );
			
			if ( expected is ulong || actual is ulong )
				return AreEqual( Convert.ToUInt64(expected), Convert.ToUInt64(actual), Convert.ToUInt64(tolerance) );
		
			if ( expected is long || actual is long )
				return AreEqual( Convert.ToInt64(expected), Convert.ToInt64(actual), Convert.ToInt64(tolerance) );
			
			if ( expected is uint || actual is uint )
				return AreEqual( Convert.ToUInt32(expected), Convert.ToUInt32(actual), Convert.ToUInt32(tolerance) );
			return AreEqual( Convert.ToInt32(expected), Convert.ToInt32(actual), Convert.ToInt32(tolerance) );
		}
		private static bool AreEqual( double expected, double actual, ref object tolerance )
		{
            if (double.IsNaN(expected) && double.IsNaN(actual))
                return true;
            // handle infinity specially since subtracting two infinite values gives 
            // NaN and the following test fails. mono also needs NaN to be handled
            // specially although ms.net could use either method.
            if (double.IsInfinity(expected) || double.IsNaN(expected) || double.IsNaN(actual))
                return expected.Equals(actual);
            if (tolerance != null)
                return Math.Abs(expected - actual) <= Convert.ToDouble(tolerance);
            if (GlobalSettings.DefaultFloatingPointTolerance > 0.0d
                && !double.IsNaN(expected) && !double.IsInfinity(expected))
            {
                tolerance = GlobalSettings.DefaultFloatingPointTolerance;
                return Math.Abs(expected - actual) <= GlobalSettings.DefaultFloatingPointTolerance;
            }
			return expected.Equals( actual );
		}
		private static bool AreEqual( decimal expected, decimal actual, decimal tolerance )
		{
			if ( tolerance > 0m )
				return Math.Abs(expected - actual) <= tolerance;
				
			return expected.Equals( actual );
		}
		private static bool AreEqual( ulong expected, ulong actual, ulong tolerance )
		{
			if ( tolerance > 0ul )
			{
				ulong diff = expected >= actual ? expected - actual : actual - expected;
				return diff <= tolerance;
			}
			return expected.Equals( actual );
		}
		private static bool AreEqual( long expected, long actual, long tolerance )
		{
			if ( tolerance > 0L )
				return Math.Abs(expected - actual) <= tolerance;
			return expected.Equals( actual );
		}
		private static bool AreEqual( uint expected, uint actual, uint tolerance )
		{
			if ( tolerance > 0 )
			{
				uint diff = expected >= actual ? expected - actual : actual - expected;
				return diff <= tolerance;
			}
				
			return expected.Equals( actual );
		}
		private static bool AreEqual( int expected, int actual, int tolerance )
		{
			if ( tolerance > 0 )
				return Math.Abs(expected - actual) <= tolerance;
				
			return expected.Equals( actual );
		}
		#endregion
		#region Numeric Comparisons 
        /// 
        /// Compare two numeric values, performing the usual numeric conversions.
        /// 
        /// The expected value
        /// The actual value
        /// 
		public static int Compare( IComparable expected, object actual )
		{
			if ( expected == null )
				throw new ArgumentException( "Cannot compare using a null reference", "expected" );
			if ( actual == null )
				throw new ArgumentException( "Cannot compare to null reference", "actual" );
			if( IsNumericType( expected ) && IsNumericType( actual ) )
			{
				if ( IsFloatingPointNumeric(expected) || IsFloatingPointNumeric(actual) )
					return Convert.ToDouble(expected).CompareTo(Convert.ToDouble(actual));
				if ( expected is decimal || actual is decimal )
					return Convert.ToDecimal(expected).CompareTo(Convert.ToDecimal(actual));
			
				if ( expected is ulong || actual is ulong )
					return Convert.ToUInt64(expected).CompareTo(Convert.ToUInt64(actual));
		
				if ( expected is long || actual is long )
					return Convert.ToInt64(expected).CompareTo(Convert.ToInt64(actual));
			
				if ( expected is uint || actual is uint )
					return Convert.ToUInt32(expected).CompareTo(Convert.ToUInt32(actual));
				return Convert.ToInt32(expected).CompareTo(Convert.ToInt32(actual));
			}
			else
				return expected.CompareTo(actual);
		}
		#endregion
		private Numerics()
		{
		}
	}
}