//============================================================================
//ZedGraph Class Library - A Flexible Line Graph/Bar Graph Library in C#
//Copyright ?2004  John Champion
//
//This library is free software; you can redistribute it and/or
//modify it under the terms of the GNU Lesser General Public
//License as published by the Free Software Foundation; either
//version 2.1 of the License, or (at your option) any later version.
//
//This library 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
//Lesser General Public License for more details.
//
//You should have received a copy of the GNU Lesser General Public
//License along with this library; if not, write to the Free Software
//Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
//=============================================================================
using System;
using System.Text;
using System.Drawing;
namespace DrawGraph
{
	/// 
	/// A class designed to simplify the process of getting the actual value for
	/// the various stacked and regular curve types
	/// 
	/// 
	///  John Champion
	///  $Revision: 3.19 $ $Date: 2007/04/16 00:03:02 $ 
	public class ValueHandler
	{
		private GraphPane _pane;
		/// 
		/// Basic constructor that saves a reference to the parent
		///  object.
		/// 
		/// The parent  object.
		/// A  flag to indicate whether or
		/// not the drawing variables should be initialized.  Initialization is not
		/// required if this is part of a ZedGraph internal draw operation (i.e., its in
		/// the middle of a call to ).  Otherwise, you should
		/// initialize to make sure the drawing variables are configured.  true to do
		/// an initialization, false otherwise.
		public ValueHandler( GraphPane pane, bool initialize )
		{
			_pane = pane;
			if ( initialize )
			{
				// just create a dummy image, which results in a full draw operation
				using ( Image image = pane.GetImage() )
				{
				}
			}
		}
		/// 
		/// Get the user scale values associate with a particular point of a
		/// particular curve.
		/// The main purpose of this method is to handle
		/// stacked bars, in which case the stacked values are returned rather
		/// than the individual data values.
		/// 
		/// A  object of interest.
		/// The zero-based point index for the point of interest.
		/// A  value representing the value
		/// for the independent axis.
		/// A  value representing the lower
		/// value for the dependent axis.
		/// A  value representing the upper
		/// value for the dependent axis.
		/// true if the data point is value, false for
		/// , invalid, etc. data.
		public bool GetValues( CurveItem curve, int iPt, out double baseVal,
							out double lowVal, out double hiVal )
		{
			return GetValues( _pane, curve, iPt, out baseVal,
									out lowVal, out hiVal );
		}
		/// 
		/// Get the user scale values associate with a particular point of a
		/// particular curve.
		/// The main purpose of this method is to handle
		/// stacked bars and lines, in which case the stacked values are returned rather
		/// than the individual data values.  However, this method works generically for any
		/// curve type.
		/// 
		/// The parent  object.
		/// A  object of interest.
		/// The zero-based point index for the point of interest.
		/// A  value representing the value
		/// for the independent axis.
		/// A  value representing the lower
		/// value for the dependent axis.
		/// A  value representing the upper
		/// value for the dependent axis.
		/// true if the data point is value, false for
		/// , invalid, etc. data.
		public static bool GetValues( GraphPane pane, CurveItem curve, int iPt,
							out double baseVal, out double lowVal, out double hiVal )
		{
			hiVal = PointPair.Missing;
			lowVal = PointPair.Missing;
			baseVal = PointPair.Missing;
			if ( curve == null || curve.Points.Count <= iPt || !curve.IsVisible )
				return false;
			Axis baseAxis = curve.BaseAxis( pane );
			Axis valueAxis = curve.ValueAxis( pane );
			if ( baseAxis is XAxis || baseAxis is X2Axis )
				baseVal = curve.Points[iPt].X;
			else
				baseVal = curve.Points[iPt].Y;
			// is it a stacked bar type?
			if ( curve is BarItem && ( pane._barSettings.Type == BarType.Stack ||
						pane._barSettings.Type == BarType.PercentStack ) )
			{
				double positiveStack = 0;
				double negativeStack = 0;
				double curVal;
				// loop through all the curves, summing up the values to get a total (only
				// for the current ordinal position iPt)
				foreach ( CurveItem tmpCurve in pane.CurveList )
				{
					// Sum the value for the current curve only if it is a bar
					if ( tmpCurve.IsBar && tmpCurve.IsVisible )
					{
						curVal = PointPair.Missing;
						// For non-ordinal curves, find a matching base value (must match exactly)
						if ( curve.IsOverrideOrdinal || !baseAxis._scale.IsAnyOrdinal )
						{
							IPointList points = tmpCurve.Points;
							for ( int i=0; i= 0 )
							{
								lowVal = positiveStack;
								hiVal = ( curVal == PointPair.Missing || positiveStack == PointPair.Missing ) ?
										PointPair.Missing : positiveStack + curVal;
							}
							// otherwise, use the negative stack
							else
							{
								hiVal = negativeStack;
								lowVal = ( curVal == PointPair.Missing || negativeStack == PointPair.Missing ) ?
										PointPair.Missing : negativeStack + curVal;
							}
						}
						// Add all positive values to the positive stack, and negative values to the
						// negative stack
						if ( curVal >= 0 )
							positiveStack = ( curVal == PointPair.Missing || positiveStack == PointPair.Missing ) ?
										PointPair.Missing : positiveStack + curVal;
						else
							negativeStack = ( curVal == PointPair.Missing || negativeStack == PointPair.Missing ) ?
										PointPair.Missing : negativeStack + curVal;
					}
				}
				// if the curve is a PercentStack type, then calculate the percent for this bar
				// based on the total height of the stack
				if ( pane._barSettings.Type == BarType.PercentStack &&
							hiVal != PointPair.Missing && lowVal != PointPair.Missing )
				{
					// Use the total magnitude of the positive plus negative bar stacks to determine
					// the percentage value
					positiveStack += Math.Abs( negativeStack );
					// just to avoid dividing by zero...
					if ( positiveStack != 0 )
					{
						// calculate the percentage values
						lowVal = lowVal / positiveStack * 100.0;
						hiVal = hiVal / positiveStack * 100.0;
					}
					else
					{
						lowVal = 0;
						hiVal = 0;
					}
				}
				if ( baseVal == PointPair.Missing || lowVal == PointPair.Missing ||
						hiVal == PointPair.Missing )
					return false;
				else
					return true;
			}
			// If the curve is a stacked line type, then sum up the values similar to the stacked bar type
			else if ( curve is LineItem && pane.LineType == LineType.Stack )
			{
				double stack = 0;
				double curVal;
				// loop through all the curves, summing up the values to get a total (only
				// for the current ordinal position iPt)
				foreach ( CurveItem tmpCurve in pane.CurveList )
				{
					// make sure the curve is a Line type
					if ( tmpCurve is LineItem && tmpCurve.IsVisible )
					{
						curVal = PointPair.Missing;
						// For non-ordinal curves, find a matching base value (must match exactly)
						if ( curve.IsOverrideOrdinal || !baseAxis._scale.IsAnyOrdinal )
						{
							IPointList points = tmpCurve.Points;
							for ( int i = 0; i < points.Count; i++ )
							{
								if ( points[i].X == baseVal )
								{
									curVal = points[i].Y;
									break;
								}
							}
						}
						// otherwise, it's an ordinal type so use the value at the same ordinal position
						else if ( iPt < tmpCurve.Points.Count )
						{
							// For line types, the Y axis is always the value axis
							curVal = tmpCurve.Points[iPt].Y;
						}
						// if the current value is missing, then the rest of the stack is missing
						if ( curVal == PointPair.Missing )
							stack = PointPair.Missing;
						// if the current curve is the target curve, save the values
						if ( tmpCurve == curve )
						{
							lowVal = stack;
//							if ( curVal < 0 && stack == 0 )
//							{
//								stack = curVal;
//								lowVal = curVal;
//								hiVal = curVal;
//							}
//							else
								hiVal = ( curVal == PointPair.Missing || stack == PointPair.Missing ) ?
									PointPair.Missing : stack + curVal;
						}
						// sum all the curves to a single total.  This includes both positive and
						// negative values (unlike the bar stack type).
						stack = ( curVal == PointPair.Missing || stack == PointPair.Missing ) ?
								PointPair.Missing : stack + curVal;
					}
				}
				
				if ( baseVal == PointPair.Missing || lowVal == PointPair.Missing ||
					hiVal == PointPair.Missing )
					return false;
				else
					return true;
			}
			// otherwise, the curve is not a stacked type (not a stacked bar or stacked line)
			else
			{
				if ( curve is BarItem && pane._barSettings.Type != BarType.ClusterHiLow )
					lowVal = 0;
				else
					lowVal = curve.Points[iPt].LowValue;
				if ( baseAxis is XAxis || baseAxis is X2Axis )
					hiVal = curve.Points[iPt].Y;
				else
					hiVal = curve.Points[iPt].X;
			}
			// Special Exception: Bars on log scales should always plot from the Min value upwards,
			// since they can never be zero
			if ( curve is BarItem && valueAxis._scale.IsLog && lowVal == 0 )
				lowVal = valueAxis._scale._min;
			if ( baseVal == PointPair.Missing || hiVal == PointPair.Missing ||
					( lowVal == PointPair.Missing && ( curve is ErrorBarItem ||
						curve is HiLowBarItem ) ) )
				return false;
			else
				return true;
		}
		/// 
		/// Calculate the user scale position of the center of the specified bar, using the
		///  as specified by .  This method is
		/// used primarily by the
		///  method in order to
		/// determine the bar "location," which is defined as the center of the top of the individual bar.
		/// 
		/// The  representing the
		/// bar of interest.
		/// The width of each individual bar. This can be calculated using
		/// the  method.
		/// The cluster number for the bar of interest.  This is the ordinal
		/// position of the current point.  That is, if a particular  has
		/// 10 points, then a value of 3 would indicate the 4th point in the data array.
		/// The actual independent axis value for the bar of interest.
		/// The ordinal position of the  of interest.
		/// That is, the first bar series is 0, the second is 1, etc.  Note that this applies only
		/// to the bars.  If a graph includes both bars and lines, then count only the bars.
		/// A user scale value position of the center of the bar of interest.
		public double BarCenterValue( CurveItem curve, float barWidth, int iCluster,
										  double val, int iOrdinal )
		{
			Axis baseAxis = curve.BaseAxis( _pane );
			if ( curve is ErrorBarItem || curve is HiLowBarItem ||
					curve is OHLCBarItem || curve is JapaneseCandleStickItem )
			{
				if ( baseAxis._scale.IsAnyOrdinal && iCluster >= 0 && !curve.IsOverrideOrdinal )
					return (double) iCluster + 1.0;
				else
					return val;
			}
			else
			{
				float clusterWidth = _pane._barSettings.GetClusterWidth();
				float clusterGap = _pane._barSettings.MinClusterGap * barWidth;
				float barGap = barWidth * _pane._barSettings.MinBarGap;
				if ( ( curve.IsBar && !( _pane._barSettings.Type == BarType.Cluster ||
								_pane._barSettings.Type == BarType.ClusterHiLow ) ) )
					iOrdinal = 0;
				float centerPix = baseAxis.Scale.Transform( curve.IsOverrideOrdinal, iCluster, val )
					- clusterWidth / 2.0F + clusterGap / 2.0F +
					iOrdinal * ( barWidth + barGap ) + 0.5F * barWidth;
				return baseAxis.Scale.ReverseTransform( centerPix );
			}
		}
	}
}