//============================================================================ //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 ); } } } }