Пример #1
0
        /// <summary>
        /// Get the user scale values associate with a particular point of a particular curve.
        /// </summary>
        /// <remarks>
        /// 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.
        /// </remarks>
        /// <param name="pane">
        /// The parent <see cref="GraphPane"/> object.
        /// </param>
        /// <param name="curve">
        /// A <see cref="CurveItem"/> object of interest.
        /// </param>
        /// <param name="iPt">
        /// The zero-based point index for the point of interest.
        /// </param>
        /// <param name="baseVal">
        /// A <see cref="double"/> value representing the value for the independent axis.
        /// </param>
        /// <param name="lowVal">
        /// A <see cref="double"/> value representing the lower value for the dependent axis.
        /// </param>
        /// <param name="hiVal">
        /// A <see cref="double"/> value representing the upper value for the dependent axis.
        /// </param>
        /// <returns>
        /// true if the data point is value, false for
        /// <see cref="PointPairBase.Missing"/>, invalid, etc. data.
        /// </returns>
        public static bool GetValues(GraphPane pane, CurveItem curve, int iPt, out double baseVal, out double lowVal, out double hiVal)
        {
            hiVal = PointPairBase.Missing;
            lowVal = PointPairBase.Missing;
            baseVal = PointPairBase.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 = PointPairBase.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 ((baseAxis is XAxis || baseAxis is X2Axis) && points[i].X == baseVal)
                                {
                                    curVal = points[i].Y;
                                    break;
                                }

                                if (!(baseAxis is XAxis || baseAxis is X2Axis) && points[i].Y == baseVal)
                                {
                                    curVal = points[i].X;
                                    break;
                                }
                            }
                        }

                        // otherwise, it's an ordinal type so use the value at the same ordinal position
                        else if (iPt < tmpCurve.Points.Count)
                        {
                            // Get the value for the appropriate value axis
                            if (baseAxis is XAxis || baseAxis is X2Axis)
                            {
                                curVal = tmpCurve.Points[iPt].Y;
                            }
                            else
                            {
                                curVal = tmpCurve.Points[iPt].X;
                            }
                        }

                        // If it's a missing value, skip it
                        if (curVal == PointPairBase.Missing)
                        {
                            positiveStack = PointPairBase.Missing;
                            negativeStack = PointPairBase.Missing;
                        }

                        // the current curve is the target curve, save the summed values for later
                        if (tmpCurve == curve)
                        {
                            // if the value is positive, use the positive stack
                            if (curVal >= 0)
                            {
                                lowVal = positiveStack;
                                hiVal = (curVal == PointPairBase.Missing || positiveStack == PointPairBase.Missing)
                                            ? PointPairBase.Missing
                                            : positiveStack + curVal;
                            }

                            // otherwise, use the negative stack
                            else
                            {
                                hiVal = negativeStack;
                                lowVal = (curVal == PointPairBase.Missing || negativeStack == PointPairBase.Missing)
                                             ? PointPairBase.Missing
                                             : negativeStack + curVal;
                            }
                        }

                        // Add all positive values to the positive stack, and negative values to the
                        // negative stack
                        if (curVal >= 0)
                        {
                            positiveStack = (curVal == PointPairBase.Missing || positiveStack == PointPairBase.Missing)
                                                ? PointPairBase.Missing
                                                : positiveStack + curVal;
                        }
                        else
                        {
                            negativeStack = (curVal == PointPairBase.Missing || negativeStack == PointPairBase.Missing)
                                                ? PointPairBase.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 != PointPairBase.Missing && lowVal != PointPairBase.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 == PointPairBase.Missing || lowVal == PointPairBase.Missing || hiVal == PointPairBase.Missing)
                {
                    return false;
                }

                return true;
            }

                // If the curve is a stacked line type, then sum up the values similar to the stacked bar type
            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 = PointPairBase.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 == PointPairBase.Missing)
                        {
                            stack = PointPairBase.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 == PointPairBase.Missing || stack == PointPairBase.Missing) ? PointPairBase.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 == PointPairBase.Missing || stack == PointPairBase.Missing) ? PointPairBase.Missing : stack + curVal;
                    }
                }

                if (baseVal == PointPairBase.Missing || lowVal == PointPairBase.Missing || hiVal == PointPairBase.Missing)
                {
                    return false;
                }

                return true;
            }

                // otherwise, the curve is not a stacked type (not a stacked bar or stacked line)
            if ((!(curve is HiLowBarItem)) && (!(curve is ErrorBarItem)))
            {
                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 == PointPairBase.Missing || hiVal == PointPairBase.Missing
                || (lowVal == PointPairBase.Missing && (curve is ErrorBarItem || curve is HiLowBarItem)))
            {
                return false;
            }

            return true;
        }
Пример #2
0
        /// <summary>
        /// Calculate the user scale position of the center of the specified bar, using the
        /// <see cref="Axis"/> as specified by <see cref="BarSettings.Base"/>.  This method is used primarily by the
        /// <see cref="GraphPane.FindNearestPoint(PointF,out CurveItem,out int)"/> method in order to determine the bar "location," which is defined as the
        /// center of the top of the individual bar.
        /// </summary>
        /// <param name="curve">
        /// The <see cref="CurveItem"/> representing the bar of interest.
        /// </param>
        /// <param name="barWidth">
        /// The width of each individual bar. This can be calculated using the <see cref="CurveItem.GetBarWidth"/> method.
        /// </param>
        /// <param name="iCluster">
        /// The cluster number for the bar of interest.  This is the ordinal position of the current point.  That is, if a particular <see cref="CurveItem"/>
        /// has 10 points, then a value of 3 would indicate the 4th point in the data array.
        /// </param>
        /// <param name="val">
        /// The actual independent axis value for the bar of interest.
        /// </param>
        /// <param name="iOrdinal">
        /// The ordinal position of the <see cref="CurveItem"/> 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.
        /// </param>
        /// <returns>
        /// A user scale value position of the center of the bar of interest.
        /// </returns>
        public double BarCenterValue(CurveItem curve, float barWidth, int iCluster, double val, int iOrdinal)
        {
            Axis baseAxis = curve.BaseAxis(this._pane);
            if (curve is ErrorBarItem || curve is HiLowBarItem || curve is OHLCBarItem || curve is JapaneseCandleStickItem)
            {
                if (baseAxis._scale.IsAnyOrdinal && iCluster >= 0 && !curve.IsOverrideOrdinal)
                {
                    return iCluster + 1.0;
                }

                return val;
            }

            float clusterWidth = this._pane._barSettings.GetClusterWidth();
            float clusterGap = this._pane._barSettings.MinClusterGap * barWidth;
            float barGap = barWidth * this._pane._barSettings.MinBarGap;

            if (curve.IsBar && this._pane._barSettings.Type != BarType.Cluster)
            {
                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);
        }