#region PDFsharp Charting - A .NET charting library based on PDFsharp // // Authors: // Niklas Schneider // // Copyright (c) 2005-2017 empira Software GmbH, Cologne Area (Germany) // // http://www.pdfsharp.com // http://sourceforge.net/projects/pdfsharp // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. #endregion using System; using PdfSharp.Drawing; namespace PdfSharp.Charting.Renderers { /// /// Represents a Y axis renderer used for charts of type Bar2D. /// internal class HorizontalYAxisRenderer : YAxisRenderer { /// /// Initializes a new instance of the HorizontalYAxisRenderer class with the /// specified renderer parameters. /// internal HorizontalYAxisRenderer(RendererParameters parms) : base(parms) { } /// /// Returns a initialized rendererInfo based on the Y axis. /// internal override RendererInfo Init() { Chart chart = (Chart)_rendererParms.DrawingItem; XGraphics gfx = _rendererParms.Graphics; AxisRendererInfo yari = new AxisRendererInfo(); yari._axis = chart._yAxis; InitScale(yari); if (yari._axis != null) { ChartRendererInfo cri = (ChartRendererInfo)_rendererParms.RendererInfo; InitTickLabels(yari, cri.DefaultFont); InitAxisTitle(yari, cri.DefaultFont); InitAxisLineFormat(yari); InitGridlines(yari); } return yari; } /// /// Calculates the space used for the Y axis. /// internal override void Format() { AxisRendererInfo yari = ((ChartRendererInfo)_rendererParms.RendererInfo).yAxisRendererInfo; if (yari._axis != null) { XGraphics gfx = _rendererParms.Graphics; XSize size = new XSize(0, 0); // height of all ticklabels double yMin = yari.MinimumScale; double yMax = yari.MaximumScale; double yMajorTick = yari.MajorTick; double lineHeight = Double.MinValue; XSize labelSize = new XSize(0, 0); for (double y = yMin; y <= yMax; y += yMajorTick) { string str = y.ToString(yari.TickLabelsFormat); labelSize = gfx.MeasureString(str, yari.TickLabelsFont); size.Width += labelSize.Width; size.Height = Math.Max(labelSize.Height, size.Height); lineHeight = Math.Max(lineHeight, labelSize.Width); } // add space for tickmarks size.Height += yari.MajorTickMarkWidth * 1.5; // Measure axis title XSize titleSize = new XSize(0, 0); if (yari._axisTitleRendererInfo != null) { RendererParameters parms = new RendererParameters(); parms.Graphics = gfx; parms.RendererInfo = yari; AxisTitleRenderer atr = new AxisTitleRenderer(parms); atr.Format(); titleSize.Height = yari._axisTitleRendererInfo.Height; titleSize.Width = yari._axisTitleRendererInfo.Width; } yari.Height = size.Height + titleSize.Height; yari.Width = Math.Max(size.Width, titleSize.Width); yari.InnerRect = yari.Rect; yari.LabelSize = labelSize; } } /// /// Draws the vertical Y axis. /// internal override void Draw() { AxisRendererInfo yari = ((ChartRendererInfo)_rendererParms.RendererInfo).yAxisRendererInfo; double yMin = yari.MinimumScale; double yMax = yari.MaximumScale; double yMajorTick = yari.MajorTick; double yMinorTick = yari.MinorTick; XMatrix matrix = new XMatrix(); matrix.TranslatePrepend(-yMin, -yari.Y); matrix.Scale(yari.InnerRect.Width / (yMax - yMin), 1, XMatrixOrder.Append); matrix.Translate(yari.X, yari.Y, XMatrixOrder.Append); // Draw axis. // First draw tick marks, second draw axis. double majorTickMarkStart = 0, majorTickMarkEnd = 0, minorTickMarkStart = 0, minorTickMarkEnd = 0; GetTickMarkPos(yari, ref majorTickMarkStart, ref majorTickMarkEnd, ref minorTickMarkStart, ref minorTickMarkEnd); XGraphics gfx = _rendererParms.Graphics; LineFormatRenderer lineFormatRenderer = new LineFormatRenderer(gfx, yari.LineFormat); XPoint[] points = new XPoint[2]; if (yari.MinorTickMark != TickMarkType.None) { for (double y = yMin + yMinorTick; y < yMax; y += yMinorTick) { points[0].X = y; points[0].Y = minorTickMarkStart; points[1].X = y; points[1].Y = minorTickMarkEnd; matrix.TransformPoints(points); lineFormatRenderer.DrawLine(points[0], points[1]); } } XStringFormat xsf = new XStringFormat(); xsf.LineAlignment = XLineAlignment.Near; int countTickLabels = (int)((yMax - yMin) / yMajorTick) + 1; for (int i = 0; i < countTickLabels; ++i) { double y = yMin + yMajorTick * i; string str = y.ToString(yari.TickLabelsFormat); XSize labelSize = gfx.MeasureString(str, yari.TickLabelsFont); if (yari.MajorTickMark != TickMarkType.None) { labelSize.Height += 1.5f * yari.MajorTickMarkWidth; points[0].X = y; points[0].Y = majorTickMarkStart; points[1].X = y; points[1].Y = majorTickMarkEnd; matrix.TransformPoints(points); lineFormatRenderer.DrawLine(points[0], points[1]); } XPoint[] layoutText = new XPoint[1]; layoutText[0].X = y; layoutText[0].Y = yari.Y + 1.5 * yari.MajorTickMarkWidth; matrix.TransformPoints(layoutText); layoutText[0].X -= labelSize.Width / 2; // Center text vertically. gfx.DrawString(str, yari.TickLabelsFont, yari.TickLabelsBrush, layoutText[0], xsf); } if (yari.LineFormat != null) { points[0].X = yMin; points[0].Y = yari.Y; points[1].X = yMax; points[1].Y = yari.Y; matrix.TransformPoints(points); if (yari.MajorTickMark != TickMarkType.None) { // yMax is at the upper side of the axis points[0].X -= yari.LineFormat.Width / 2; points[1].X += yari.LineFormat.Width / 2; } lineFormatRenderer.DrawLine(points[0], points[1]); } // Draw axis title if (yari._axisTitleRendererInfo != null) { RendererParameters parms = new RendererParameters(); parms.Graphics = gfx; parms.RendererInfo = yari; XRect rcTitle = yari.Rect; rcTitle.Height = yari._axisTitleRendererInfo.Height; rcTitle.Y += yari.Rect.Height - rcTitle.Height; yari._axisTitleRendererInfo.Rect = rcTitle; AxisTitleRenderer atr = new AxisTitleRenderer(parms); atr.Draw(); } } /// /// Calculates all values necessary for scaling the axis like minimum/maximum scale or /// minor/major tick. /// private void InitScale(AxisRendererInfo rendererInfo) { double yMin, yMax; CalcYAxis(out yMin, out yMax); FineTuneYAxis(rendererInfo, yMin, yMax); rendererInfo.MajorTickMarkWidth = DefaultMajorTickMarkWidth; rendererInfo.MinorTickMarkWidth = DefaultMinorTickMarkWidth; } /// /// Gets the top and bottom position of the major and minor tick marks depending on the /// tick mark type. /// private void GetTickMarkPos(AxisRendererInfo rendererInfo, ref double majorTickMarkStart, ref double majorTickMarkEnd, ref double minorTickMarkStart, ref double minorTickMarkEnd) { double majorTickMarkWidth = rendererInfo.MajorTickMarkWidth; double minorTickMarkWidth = rendererInfo.MinorTickMarkWidth; double y = rendererInfo.Rect.Y; switch (rendererInfo.MajorTickMark) { case TickMarkType.Inside: majorTickMarkStart = y - majorTickMarkWidth; majorTickMarkEnd = y; break; case TickMarkType.Outside: majorTickMarkStart = y; majorTickMarkEnd = y + majorTickMarkWidth; break; case TickMarkType.Cross: majorTickMarkStart = y - majorTickMarkWidth; majorTickMarkEnd = y + majorTickMarkWidth; break; //TickMarkType.None: default: majorTickMarkStart = 0; majorTickMarkEnd = 0; break; } switch (rendererInfo.MinorTickMark) { case TickMarkType.Inside: minorTickMarkStart = y - minorTickMarkWidth; minorTickMarkEnd = y; break; case TickMarkType.Outside: minorTickMarkStart = y; minorTickMarkEnd = y + minorTickMarkWidth; break; case TickMarkType.Cross: minorTickMarkStart = y - minorTickMarkWidth; minorTickMarkEnd = y + minorTickMarkWidth; break; //TickMarkType.None: default: minorTickMarkStart = 0; minorTickMarkEnd = 0; break; } } /// /// Determines the smallest and the largest number from all series of the chart. /// protected virtual void CalcYAxis(out double yMin, out double yMax) { yMin = double.MaxValue; yMax = double.MinValue; foreach (Series series in ((Chart)_rendererParms.DrawingItem).SeriesCollection) { foreach (Point point in series.Elements) { if (!double.IsNaN(point._value)) { yMin = Math.Min(yMin, point.Value); yMax = Math.Max(yMax, point.Value); } } } } } }