//-------------------------------------------------------------
// 
//   Copyright © Microsoft Corporation. All Rights Reserved.
// 
//-------------------------------------------------------------
// @owner=alexgor, deliant
//=================================================================
//  File:		Oscillator.cs
//
//  Namespace:	System.Web.UI.WebControls[Windows.Forms].Charting.Formulas
//
//	Classes:	Oscillators
//
//  Purpose:	This class is used to calculate oscillator 
//				indicators used in Technical Analyses.
//
//	Reviewed:	GS - August 7, 2002
//				AG - August 7, 2002
//
//===================================================================
using System;
using System.Globalization;
#if Microsoft_CONTROL
	namespace System.Windows.Forms.DataVisualization.Charting.Formulas
#else
	namespace System.Web.UI.DataVisualization.Charting.Formulas
#endif
{
	/// 
	/// This class is used to calculate oscillator 
	///	indicators used in Technical Analyses.
	/// 
	internal class Oscillators : PriceIndicators
	{
		#region Properties
		/// 
		/// Formula Module name
		/// 
		override public string Name			{ get{ return "Oscillators";}}
		#endregion
		#region Formulas
		/// 
		/// The Chaikin Oscillator is created by subtracting a 10 period 
		/// exponential moving average of the Accumulation/Distribution 
		/// line from a 3 period moving average of the 
		/// Accumulation/Distribution Line. 
		/// ---------------------------------------------------------
		/// Input: 
		///		- 4 Y values ( Hi, Low, Close, Volume ).
		/// Output: 
		///		- 1 Y value Chaikin Oscillator
		/// Parameters: 
		///		- Short Period for Exponential Moving average (default=3)
		///		- Int64 Period for Exponential Moving average (default=10)
		///	Extra Parameters: 
		///		- Start from First
		/// 
		/// Arrays of doubles - Input values
		/// Arrays of doubles - Output values
		/// Array of strings - Parameters
		/// Array of strings - Extra parameters
		private void ChaikinOscillator(double [][] inputValues, out double [][] outputValues, string [] parameterList, string [] extraParameterList)
		{
			// There is no enough input series
			if( inputValues.Length != 5 )
			{
				throw new ArgumentException( SR.ExceptionPriceIndicatorsFormulaRequiresFourArrays);
			}
						
			// Different number of x and y values
			CheckNumOfValues( inputValues, 4 );
			// Short Period for Exp moving average
			int shortPeriod;
            if (parameterList.Length < 1 || 
                !int.TryParse(parameterList[0], NumberStyles.Any, CultureInfo.InvariantCulture, out shortPeriod))
            {
                shortPeriod = 3;
            }
			
			// Int64 Period for Exp moving average
			int longPeriod;
            if (parameterList.Length < 2 || 
                !int.TryParse(parameterList[1], NumberStyles.Any, CultureInfo.InvariantCulture, out longPeriod))
            {
                longPeriod = 10;
            }
			if( shortPeriod > longPeriod || longPeriod <= 0 || shortPeriod <= 0 )
			{
                throw new ArgumentException(SR.ExceptionOscillatorObjectInvalidPeriod);
			}
			// Starting average from the first data point or after period.
			bool startFromFirst = bool.Parse( extraParameterList[0] );
			VolumeIndicators volume = new VolumeIndicators();
			double [][] outputDistribution = new double [2][];
			// Accumulation Distribution
			volume.AccumulationDistribution( inputValues, out outputDistribution );
			double [] ExpAvgDistribution;
			// Exponential Moving average of Accumulation Distribution
			ExponentialMovingAverage(outputDistribution[1],out ExpAvgDistribution,longPeriod,startFromFirst);
			double [] ExpAvg;
			// Exponential Moving average of close
			ExponentialMovingAverage(outputDistribution[1],out ExpAvg,shortPeriod,startFromFirst);
			outputValues = new double [2][];
			int period = Math.Min(ExpAvg.Length,ExpAvgDistribution.Length);
			outputValues[0] = new double [period];
			outputValues[1] = new double [period];
			// Accumulation Distribution
			int expIndex = 0;
			for( int index = inputValues[1].Length - period; index < inputValues[1].Length; index++ )
			{
				// Set X values
				outputValues[0][expIndex] = inputValues[0][index];
				// Set Y values
				if(startFromFirst)
				{
					// Number of items in all arays is the same and they are aligned by time.
					outputValues[1][expIndex] = ExpAvg[expIndex] - ExpAvgDistribution[expIndex];
				}
				else if( (expIndex + longPeriod - shortPeriod) < ExpAvg.Length)
				{
					// Number of items in MovingAverages arrays is different and requires adjustment.
					outputValues[1][expIndex] = ExpAvg[expIndex + longPeriod - shortPeriod] - ExpAvgDistribution[expIndex];
				}
				else
				{
					outputValues[1][expIndex] = Double.NaN;
				}
				expIndex++;
			}
		}
		/// 
		/// The Detrended Price Oscillator ("DPO") attempts to 
		/// eliminate the trend in prices. Detrended prices allow 
		/// you to more easily identify cycles and overbought/oversold 
		/// levels. To calculate the DPO, you specify a time period. 
		/// Cycles longer than this time period are removed from 
		/// prices, leaving the shorter-term cycles. 
		/// ---------------------------------------------------------
		/// Input: 
		///		- 1 Y value ( Close ).
		/// Output: 
		///		- 1 Y value Detrended Price Oscillator
		/// Parameters: 
		///		- Period
		/// 
		/// Arrays of doubles - Input values
		/// Arrays of doubles - Output values
		/// Array of strings - Parameters
		private void DetrendedPriceOscillator(double [][] inputValues, out double [][] outputValues, string [] parameterList)
		{
			// There is no enough input series
			if( inputValues.Length != 2 )
			{
                throw new ArgumentException(SR.ExceptionPriceIndicatorsFormulaRequiresOneArray);
			}
			
			// Different number of x and y values
			CheckNumOfValues( inputValues, 1 );
			
			// Short Period for Exp moving average
			int period;
			try
			{period = int.Parse( parameterList[0], System.Globalization.CultureInfo.InvariantCulture );}
			catch( Exception e )
			{
                if (e.Message == SR.ExceptionObjectReferenceIsNull)
                    throw new InvalidOperationException(SR.ExceptionPriceIndicatorsPeriodMissing);
				else
                    throw new InvalidOperationException(SR.ExceptionPriceIndicatorsPeriodMissing + e.Message);
			}
			if( period <= 0 )
                throw new InvalidOperationException(SR.ExceptionPeriodParameterIsNegative);
			double [] outputAverage;
			// Moving Average
			MovingAverage( inputValues[1], out outputAverage, period, false );
			
			outputValues = new double [2][];
			outputValues[0] = new double [inputValues[0].Length - period*3/2];
			outputValues[1] = new double [inputValues[1].Length - period*3/2];
			// Detrended Price Oscillator
			for( int index = 0; index < outputValues[1].Length; index++ )
			{
				// Set X values
				outputValues[0][index] = inputValues[0][index + period + period/2];
				// Set Y values
				outputValues[1][index] = inputValues[1][index + period + period/2] - outputAverage[index];
			}
		}
		/// 
		/// Chaikin's Volatility indicator compares the spread 
		/// between a security's high and low prices. 
		/// It quantifies volatility as a widening of the range 
		/// between the high and the low price. There are two ways 
		/// to interpret this measure of volatility. One method 
		/// assumes that market tops are generally accompanied by 
		/// increased volatility (as investors get nervous and 
		/// indecisive) and that the latter stages of a market 
		/// bottom are generally accompanied by decreased volatility 
		/// (as investors get bored). Another method (Mr. Chaikin's) 
		/// assumes that an increase in the Volatility indicator over 
		/// a relatively short time period indicates that a bottom is 
		/// near (e.g., a panic sell-off) and that a decrease in 
		/// volatility over a longer time period indicates an 
		/// approaching top (e.g., a mature bull market). As with 
		/// almost all experienced investors, Mr. Chaikin recommends 
		/// that you do not rely on any one indicator. He suggests 
		/// using a moving average ----ion or trading band system 
		/// to confirm this (or any) indicator.
		/// ---------------------------------------------------------
		/// Input: 
		///		- 2 Y values ( Hi, Low ).
		/// Output: 
		///		- 1 Y value Volatility Chaikins
		/// Parameters: 
		///		- Periods (default 10)- is used to specify the Shift days, By default this property is set to 10.
		///     - SignalPeriod (default 10)- is used to calculate Exponential Moving Avg of the Signal line, By default this property is set to 10.
		/// 
		/// Arrays of doubles - Input values
		/// Arrays of doubles - Output values
		/// Array of strings - Parameters
		private void VolatilityChaikins(double [][] inputValues, out double [][] outputValues, string [] parameterList)
		{
			// There is no enough input series
			if( inputValues.Length != 3 )
			{
				throw new ArgumentException( SR.ExceptionPriceIndicatorsFormulaRequiresTwoArrays);
			}
			
			// Different number of x and y values
			CheckNumOfValues( inputValues, 2 );
						
			// Period
			int period;
            if (parameterList.Length < 1 ||
                !int.TryParse(parameterList[0], NumberStyles.Any, CultureInfo.InvariantCulture, out period))
            {
                period = 10;
            }
			if( period <= 0 )
                throw new InvalidOperationException(SR.ExceptionOscillatorNegativePeriodParameter);
			// Signal Period for Exp moving average
			int signalPeriod;
            if (parameterList.Length < 2 ||
                !int.TryParse(parameterList[1], NumberStyles.Any, CultureInfo.InvariantCulture, out signalPeriod))
            {
                signalPeriod = 10;
            }
			if( signalPeriod <= 0 )
                throw new InvalidOperationException(SR.ExceptionOscillatorNegativeSignalPeriod);
			double [] outputAverage;
			double [] hiLowInput = new double[inputValues[1].Length];
			// Find Hi - Low
			for( int index = 0; index < inputValues[1].Length; index++ )
			{
				hiLowInput[index] = inputValues[1][index] - inputValues[2][index];
			}
            
			// Exponential Moving Average
			ExponentialMovingAverage( hiLowInput, out outputAverage, signalPeriod, false );
			
			outputValues = new double [2][];
			outputValues[0] = new double [outputAverage.Length - period];
			outputValues[1] = new double [outputAverage.Length - period];
			// Volatility Chaikins
			for( int index = 0; index < outputValues[1].Length; index++ )
			{
				// Set X values
				outputValues[0][index] = inputValues[0][index + period + signalPeriod - 1];
				// Set Y values
				if( outputAverage[index] != 0.0 )
					outputValues[1][index] = ( outputAverage[index + period] - outputAverage[index] ) / outputAverage[index] * 100.0;
				else
					// Div with zero error.
					outputValues[1][index] = 0.0;
			}
		}
		/// 
		/// The Volume Oscillator displays the difference between two 
		/// moving averages of a security's volume. The difference 
		/// between the moving averages can be expressed in either 
		/// points or percentages. You can use the difference between 
		/// two moving averages of volume to determine if the overall 
		/// volume trend is increasing or decreasing. When the Volume 
		/// Oscillator rises above zero, it signifies that the 
		/// shorter-term volume moving average has risen above 
		/// the longer-term volume moving average, and thus, that 
		/// the short-term volume trend is higher (i.e., more volume) 
		/// than the longer-term volume trend.
		/// ---------------------------------------------------------
		/// Input: 
		///		- 1 Y values ( Volume ).
		/// Output: 
		///		- 1 Y value VolumeOscillator
		/// Parameters: 
		///		- ShortPeriod (Default 5)= is used to configure the short period.
		///		- LongPeriod (Default 10)= is used to configure the Int64 period.
		///		- Percentage (Default true)= The Volume Oscillator can display the difference between the two moving averages as either points or percentages.
		/// 
		/// Arrays of doubles - Input values
		/// Arrays of doubles - Output values
		/// Array of strings - Parameters
		private void VolumeOscillator(double [][] inputValues, out double [][] outputValues, string [] parameterList)
		{
			// There is no enough input series
			if( inputValues.Length != 2 )
			{
                throw new ArgumentException(SR.ExceptionPriceIndicatorsFormulaRequiresOneArray);
			}
			
			// Different number of x and y values
			CheckNumOfValues( inputValues, 1 );
						
			// Short Period
			int shortPeriod;
            if (parameterList.Length < 1 || 
                !int.TryParse(parameterList[0], NumberStyles.Any, CultureInfo.InvariantCulture, out shortPeriod))
            {
                shortPeriod = 5;
            }
			// Int64 Period
			int longPeriod;
            if (parameterList.Length < 2 || 
                !int.TryParse(parameterList[1], NumberStyles.Any, CultureInfo.InvariantCulture, out longPeriod))
            {
                longPeriod = 10;
            }
			if( shortPeriod > longPeriod || longPeriod <= 0 || shortPeriod <= 0 )
                throw new ArgumentException(SR.ExceptionOscillatorObjectInvalidPeriod);
			// percentage
			bool percentage;
            if (parameterList.Length < 3 || 
                !bool.TryParse(parameterList[2], out percentage))
            {
                percentage = true;
            }
			
			double [] shortAverage;
			double [] longAverage;
			// Find Short moving average
			MovingAverage( inputValues[1], out shortAverage, shortPeriod, false );
			// Find Int64 moving average
			MovingAverage( inputValues[1], out longAverage, longPeriod, false );
			outputValues = new double [2][];
			outputValues[0] = new double [longAverage.Length];
			outputValues[1] = new double [longAverage.Length];
			
			// Volume Oscillator
			for( int index = 0; index < longAverage.Length; index++ )
			{
				// Set X values
				outputValues[0][index] = inputValues[0][index + longPeriod-1];
				// Set Y values
				outputValues[1][index] = shortAverage[index + shortPeriod] - longAverage[index];
				// RecalculateAxesScale difference in %
				if( percentage )
				{
					// Div by zero error.
					if( longAverage[index] == 0.0 )
						outputValues[1][index]=0.0;
					else
						outputValues[1][index] = outputValues[1][index] / shortAverage[index + shortPeriod] * 100;
				}
			}
		}
		/// 
		/// The Stochastic Indicator is based on the observation that 
		/// as prices increase, closing prices tend to accumulate ever 
		/// closer to the highs for the period. Conversely, as prices 
		/// decrease, closing prices tend to accumulate ever closer to 
		/// the lows for the period. Trading decisions are made with 
		/// respect to divergence between % of "D" (one of the two 
		/// lines generated by the study) and the item's price. For 
		/// example, when a commodity or stock makes a high, reacts, 
		/// and subsequently moves to a higher high while corresponding 
		/// peaks on the % of "D" line make a high and then a lower 
		/// high, a bearish divergence is indicated. When a commodity 
		/// or stock has established a new low, reacts, and moves to a 
		/// lower low while the corresponding low points on the % of 
		/// "D" line make a low and then a higher low, a bullish 
		/// divergence is indicated. Traders act upon this divergence 
		/// when the other line generated by the study (K) crosses on 
		/// the right-hand side of the peak of the % of "D" line in the 
		/// case of a top, or on the right-hand side of the low point 
		/// of the % of "D" line in the case of a bottom. The Stochastic 
		/// Oscillator is displayed as two lines. The main line is 
		/// called "%K." The second line, called "%D," is a moving 
		/// average of %K. 
		/// ---------------------------------------------------------
		/// Input: 
		///		- 3 Y values ( Hi, Low, Close ).
		/// Output: 
		///		- 2 Y value ( %K, %D )
		/// Parameters: 
		///		- PeriodD (Default 10) = is used for %D calculation as SMA of %K.
		///     - PeriodK (Default 10) = is used to calculate %K.
		/// 
		/// Arrays of doubles - Input values
		/// Arrays of doubles - Output values
		/// Array of strings - Parameters
		internal void StochasticIndicator(double [][] inputValues, out double [][] outputValues, string [] parameterList)
		{
			// There is no enough input series
			if( inputValues.Length != 4 )
			{
				throw new ArgumentException( SR.ExceptionPriceIndicatorsFormulaRequiresThreeArrays);
			}
			
			// Different number of x and y values
			CheckNumOfValues( inputValues, 3 );
			
			// PeriodD for moving average
			int periodD;
            if (parameterList.Length < 2 || 
                !int.TryParse(parameterList[1], NumberStyles.Any, CultureInfo.InvariantCulture, out periodD))
            {
                periodD = 10;
            }
			if( periodD <= 0 )
                throw new InvalidOperationException(SR.ExceptionPeriodParameterIsNegative);
			// PeriodK for moving average
			int periodK;
            if (parameterList.Length < 1 || 
                !int.TryParse(parameterList[0], NumberStyles.Any, CultureInfo.InvariantCulture, out periodK))
            {
                periodK = 10;
            }
			if( periodK <= 0 )
                throw new InvalidOperationException(SR.ExceptionPeriodParameterIsNegative);
			// Output arrays
			outputValues = new double [3][];
			// X
			outputValues[0] = new double [inputValues[0].Length - periodK - periodD + 2];
			// K%
			outputValues[1] = new double [inputValues[0].Length - periodK - periodD + 2];
			// D%
			outputValues[2] = new double [inputValues[0].Length - periodK - periodD + 2];
			double [] K = new double [inputValues[0].Length - periodK + 1];
			// Find K%
			for( int index = periodK - 1; index < inputValues[0].Length; index++ )
			{
				// Find Lowest Low and Highest High
				double minLow = double.MaxValue;
				double maxHi = double.MinValue;
				for( int indexHL = index - periodK + 1; indexHL <= index; indexHL++ )
				{
					if( minLow > inputValues[2][indexHL] )
						minLow = inputValues[2][indexHL];
					if( maxHi < inputValues[1][indexHL] )
						maxHi = inputValues[1][indexHL];
				}
				// Find K%
				K[index - periodK + 1] = ( inputValues[3][index] - minLow ) / ( maxHi - minLow ) * 100;
				// Set X and Y K output
				if( index >= periodK + periodD - 2 )
				{
					outputValues[0][index - periodK - periodD + 2] = inputValues[0][index];
					outputValues[1][index - periodK - periodD + 2] = K[index - periodK + 1];
				}
			}
			// Find D%
			MovingAverage( K, out outputValues[2], periodD, false );
		}
		/// 
		/// Williams’ %R (pronounced "percent R") is a momentum 
		/// indicator that measures overbought/oversold levels. 
		/// Williams’ %R was developed by Larry Williams. The 
		/// interpretation of Williams' %R is very similar to that 
		/// of the Stochastic Oscillator except that %R is plotted 
		/// upside-down and the Stochastic Oscillator has internal 
		/// smoothing. To display the Williams’ %R indicator on an 
		/// upside-down scale, it is usually plotted using negative 
		/// values (e.g., -20%). Readings in the range of 80 to 100% 
		/// indicate that the security is oversold while readings in 
		/// the 0 to 20% range suggest that it is overbought.
		/// As with all overbought/oversold indicators, it is best to 
		/// wait for the security's price to change direction before 
		/// placing your trades. For example, if an overbought/oversold 
		/// indicator (such as the Stochastic Oscillator or Williams' 
		/// %R) is showing an overbought condition, it is wise to wait 
		/// for the security's price to turn down before selling the 
		/// security. (The MovingAverageConvergenceDivergence is a good indicator to monitor change 
		/// in a security's price.) It is not unusual for 
		/// overbought/oversold indicators to remain in an 
		/// overbought/oversold condition for a long time period as 
		/// the security's price continues to climb/fall. Selling 
		/// simply because the security appears overbought may take 
		/// you out of the security long before its price shows signs 
		/// of deterioration.
		/// ---------------------------------------------------------
		/// Input: 
		///		- 3 Y values ( Hi, Low, Close ).
		/// Output: 
		///		- 2 Y value ( %R )
		/// Parameters: 
		///		- Period (Default 14) = is used to configure the number of periods to calculate the WilliamsR
		/// 
		/// Arrays of doubles - Input values
		/// Arrays of doubles - Output values
		/// Array of strings - Parameters
		internal void WilliamsR(double [][] inputValues, out double [][] outputValues, string [] parameterList)
		{
			// There is no enough input series
			if( inputValues.Length != 4 )
				throw new ArgumentException( SR.ExceptionPriceIndicatorsFormulaRequiresThreeArrays);
			
			// Different number of x and y values
			CheckNumOfValues( inputValues, 3 );
			
			// PeriodD for moving average
			int period;
            if (parameterList.Length < 1 || 
                !int.TryParse(parameterList[0], NumberStyles.Any, CultureInfo.InvariantCulture, out period))
            {
                period = 14;
            }
			if( period <= 0 )
                throw new InvalidOperationException(SR.ExceptionPeriodParameterIsNegative);
			// Output arrays
			outputValues = new double [2][];
			// X
			outputValues[0] = new double [inputValues[0].Length - period + 1];
			// R%
			outputValues[1] = new double [inputValues[0].Length - period + 1];
			// Find R%
			for( int index = period - 1; index < inputValues[0].Length; index++ )
			{
				// Find Lowest Low and Highest High
				double minLow = double.MaxValue;
				double maxHi = double.MinValue;
				for( int indexHL = index - period + 1; indexHL <= index; indexHL++ )
				{
					if( minLow > inputValues[2][indexHL] )
						minLow = inputValues[2][indexHL];
					if( maxHi < inputValues[1][indexHL] )
						maxHi = inputValues[1][indexHL];
				}
				// Set X value
				outputValues[0][index - period + 1] = inputValues[0][index];
				// Find R%
				outputValues[1][index - period + 1] = ( maxHi - inputValues[3][index] ) / ( maxHi - minLow ) * (-100.0);
			}
		}
		#endregion
		#region Methods
		/// 
		/// Constructor
		/// 
		public Oscillators()
		{
		}
        /// 
        /// The first method in the module, which converts a formula 
        /// name to the corresponding private method.
        /// 
        /// String which represent a formula name
        /// Arrays of doubles - Input values
        /// Arrays of doubles - Output values
        /// Array of strings - Formula parameters
        /// Array of strings - Extra Formula parameters from DataManipulator object
        /// Array of strings - Used for Labels. Description for output results.
		override public void Formula( string formulaName, double [][] inputValues, out double [][] outputValues, string [] parameterList, string [] extraParameterList, out string [][] outLabels )
		{
			string name;
			outputValues = null;
			// Not used for these formulas.
			outLabels = null;
			name = formulaName.ToUpper(System.Globalization.CultureInfo.InvariantCulture);
			try
			{
				switch( name )
				{
					case "STOCHASTICINDICATOR":
						StochasticIndicator( inputValues, out outputValues, parameterList );
						break;
					case "CHAIKINOSCILLATOR":
						ChaikinOscillator( inputValues, out outputValues, parameterList, extraParameterList );
						break;
					case "DETRENDEDPRICEOSCILLATOR":
						DetrendedPriceOscillator( inputValues, out outputValues, parameterList );
						break;
					case "VOLATILITYCHAIKINS":
						VolatilityChaikins( inputValues, out outputValues, parameterList );
						break;
					case "VOLUMEOSCILLATOR":
						VolumeOscillator( inputValues, out outputValues, parameterList );
						break;
					case "WILLIAMSR":
						WilliamsR( inputValues, out outputValues, parameterList );
						break;
					default:
						outputValues = null;
						break;
				}
			}
			catch( IndexOutOfRangeException )
			{
				throw new InvalidOperationException( SR.ExceptionFormulaInvalidPeriod( name ) );
			}
			catch( OverflowException )
			{
				throw new InvalidOperationException( SR.ExceptionFormulaNotEnoughDataPoints( name ) );
			}
		}
		#endregion
		
	}
}