Chinaunix

标题: 没事写个拆线图,做个Memo [打印本页]

作者: folklore    时间: 2016-08-20 16:36
标题: 没事写个拆线图,做个Memo
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Drawing;
  4. using System.Drawing.Drawing2D;
  5. using System.Linq;
  6. using System.Text;
  7. using System.Threading.Tasks;

  8. namespace CSGraphLinear
  9. {
  10.     public static class BitmapEx
  11.     {
  12.         public static void ToBlackAndWhite(this Bitmap bitmap)
  13.         {

  14.         }
  15.     }

  16.     public class LinearGraph:IDisposable
  17.     {
  18.         public LinearGraph(int width, int height)
  19.         {
  20.             this._bm = new Bitmap(width, height);
  21.             this._g = Graphics.FromImage(this._bm);

  22.             this._lines = new List<KeyValuePair<System.Windows.Point[], object[]>>();
  23.             this.XAxisLables = new List<KeyValuePair<double, string>>();
  24.             this.YAxisLables = new List<KeyValuePair<double, string>>();
  25.             this._Labels = new List<KeyValuePair<System.Windows.Point, string>>();

  26.             this.Margin = new Rectangle(5, 5, 5, 5);
  27.             this.BackgroundColor = Color.White;
  28.             this.AxisColor = Color.Black;
  29.             this.AxisWidth = 1.0D;
  30.         }

  31.         public void Resize(int width, int height)
  32.         {
  33.             if (width > 0 && height > 0)
  34.             {
  35.                 this._g.Dispose();
  36.                 this._bm.Dispose();
  37.                 this._bm = new Bitmap(width, height);
  38.                 this._g = Graphics.FromImage(this._bm);

  39.                 this.Draw();
  40.             }
  41.         }

  42.         public void Draw()
  43.         {
  44.             // -- Clear background
  45.             this._g.Clear(this.BackgroundColor);

  46.             // -- Search max/min value of lines points.
  47.             double xMin = double.MaxValue,
  48.                 xMax = double.MinValue,
  49.                 yMin = double.MaxValue,
  50.                 yMax = double.MinValue;
  51.             {
  52.                 foreach (var line in this._lines)
  53.                 {
  54.                     foreach (var pt in line.Key)
  55.                     {
  56.                         if (pt.X > xMax)
  57.                         {
  58.                             xMax = pt.X;
  59.                         }
  60.                         if (pt.X < xMin)
  61.                         {
  62.                             xMin = pt.X;
  63.                         }
  64.                         if (pt.Y > yMax)
  65.                         {
  66.                             yMax = pt.Y;
  67.                         }
  68.                         if (pt.Y < yMin)
  69.                         {
  70.                             yMin = pt.Y;
  71.                         }
  72.                     }//end for each point
  73.                 }//end foreach line
  74.             }

  75.             do
  76.             {
  77.                 // -- Check the size
  78.                 if(this._bm.Width <=this.Margin.Left +this.Margin.Right ||
  79.                 this._bm.Height <= this.Margin.Top + this.Margin.Bottom) break;

  80.                 // -- Calculate the scale
  81.                 double xScale = (this._bm.Width - this.Margin.Left - this.Margin.Right) / (xMax - xMin);
  82.                 double yScale = (this._bm.Height - this.Margin.Top - this.Margin.Bottom) / (yMax - yMin);

  83.                 // -- Draw XAxis
  84.                 using (Pen p = new Pen(this.AxisColor, (float)this.AxisWidth))
  85.                 {
  86.                     this._g.DrawLine(p, this.Margin.Left, 0, this.Margin.Left, this._bm.Height);
  87.                     this._g.DrawLine(p, 0, this._bm.Height - this.Margin.Bottom, this._bm.Width, this._bm.Height - this.Margin.Bottom);
  88.                 }

  89.                 if(xMax <=xMin ||yMax <=yMin) break;

  90.                 // -- Draw scales in X/Y axises
  91.                 {   const double nAxisSplit =6.0D;
  92.                     Func<double, double> ToAlignedStepWidth = delegate(double src)
  93.                     {
  94.                         double srcCpy = src;
  95.                         double rsToAlignedStepWidth;
  96.                         double dScale = 1.0D;
  97.                         if (srcCpy >= 1.0D)
  98.                         {
  99.                             while (srcCpy > 1.0D)
  100.                             {
  101.                                 dScale *= 10.0D;
  102.                                 srcCpy = src / dScale;
  103.                             }
  104.                             rsToAlignedStepWidth = Math.Ceiling(src / dScale) * dScale;
  105.                             if ((rsToAlignedStepWidth - src) / rsToAlignedStepWidth < 0.3D)
  106.                             {
  107.                                 rsToAlignedStepWidth = Math.Floor(src / dScale) * dScale;
  108.                             }
  109.                         }
  110.                         else
  111.                         {
  112.                             while (srcCpy < 1.0D)
  113.                             {
  114.                                 dScale /= 10.0D;
  115.                                 srcCpy = src / dScale;
  116.                             }
  117.                             rsToAlignedStepWidth = Math.Ceiling(src / dScale) * dScale;
  118.                             if ((rsToAlignedStepWidth - src) / rsToAlignedStepWidth < 0.3D)
  119.                             {
  120.                                 rsToAlignedStepWidth = Math.Floor(src / dScale) * dScale;
  121.                             }
  122.                         }

  123.                         return rsToAlignedStepWidth;
  124.                     };

  125.                     double xStepLength = ToAlignedStepWidth((xMax - xMin) / nAxisSplit);
  126.                     double yStepLength = ToAlignedStepWidth((yMax - yMin) / nAxisSplit);

  127.                     if (xStepLength > xMax - xMin) xStepLength /= 5.0D;
  128.                     if ((xMax - xMin) / xStepLength < 3.0D) xStepLength /= 2.0D;
  129.                     if (yStepLength > yMax - yMin) yStepLength /= 5.0D;
  130.                     if ((yMax - yMin) / yStepLength < 3.0D) yStepLength /= 2.0D;

  131.                     // -- Draw X/Y axis scale
  132.                     {
  133.                         using (Brush br = new SolidBrush(this.AxisColor))
  134.                         {
  135.                             using (Pen p = new Pen(this.AxisColor))
  136.                             {
  137.                                 {   // -- X axis
  138.                                     int y = this._bm.Height - this.Margin.Bottom;
  139.                                     for (double xStart = Math.Ceiling(xMin / xStepLength) * xStepLength; xStart < xMax; xStart += xStepLength)
  140.                                     {
  141.                                         int x = (int)(xScale * (xStart - xMin) + this.Margin.Left);
  142.                                         this._g.DrawLine(p, x, y - 1 - (int)this.AxisWidth, x, y + 1);
  143.                                         if (this.XAxisLables.Count == 0)
  144.                                         {   // -- If the X axis labels specified, not draw the scale
  145.                                             this._g.DrawString(xStart.ToString(), SystemFonts.DefaultFont, br, new Point(x, y));
  146.                                         }
  147.                                     }

  148.                                     // -- Draw X axis labels
  149.                                     foreach (var lbl in this.XAxisLables)
  150.                                     {
  151.                                         if (!string.IsNullOrWhiteSpace(lbl.Value))
  152.                                         {
  153.                                             int x = (int)(xScale * (lbl.Key - xMin) + this.Margin.Left);
  154.                                             this._g.DrawString(lbl.Value, SystemFonts.DefaultFont, br, new Point(x, y));
  155.                                         }
  156.                                     }
  157.                                 }

  158.                                 {   // -- Y axis
  159.                                     int x = this.Margin.Left;
  160.                                     for (double yStart = Math.Ceiling(yMin / yStepLength) * yStepLength; yStart < yMax; yStart += yStepLength)
  161.                                     {
  162.                                         int y = this._bm.Height - (int)(yScale * (yStart - yMin) + this.Margin.Bottom);

  163.                                         this._g.DrawLine(p, x - 1 - (int)this.AxisWidth, y, x + 1, y);
  164.                                         this._g.DrawString(yStart.ToString(), SystemFonts.DefaultFont, br, new Point(x, y));

  165.                                         if (IsDrawHorizontalLine)
  166.                                         {
  167.                                             using (Pen pVertical = new Pen(this.AxisColor) { DashStyle = DashStyle.Dash })
  168.                                             {
  169.                                                 this._g.DrawLine(pVertical, x + 1, y, this._bm.Width, y);
  170.                                             }
  171.                                         }
  172.                                     }
  173.                                 }
  174.                             }
  175.                         }
  176.                     }
  177.                 }

  178.                 // -- Draw lines
  179.                 foreach (var line in this._lines)
  180.                 {
  181.                     Point[] pts =new Point[line.Key.Length];
  182.                     for (int iPt = 0; iPt < line.Key.Length; iPt++)
  183.                     {
  184.                         int x = (int)(xScale * (line.Key[iPt].X - xMin) + this.Margin.Left);
  185.                         int y = this._bm.Height -(int)(yScale * (line.Key[iPt].Y-yMin) + this.Margin.Bottom);
  186.                         pts[iPt] = new Point(x,y);
  187.                     }
  188.                     using (Pen p = new Pen((Color)line.Value[(int)LineStyle.LineColor]) {
  189.                         DashStyle = (DashStyle)line.Value[(int)LineStyle.PenStyle],
  190.                         Width = (float)Convert.ToDouble(line.Value[(int)LineStyle.LineWidth])
  191.                     })
  192.                     {
  193.                         this._g.DrawLines(p, pts);
  194.                     }
  195.                 }//end foreach: line

  196.                 // -- Draw lables
  197.                 using (Brush br = new SolidBrush(this.AxisColor))
  198.                 {
  199.                     foreach (var kv in this._Labels)
  200.                     {
  201.                         if (!string.IsNullOrWhiteSpace(kv.Value))
  202.                         {
  203.                             int x = (int)(xScale * (kv.Key.X - xMin) + this.Margin.Left);
  204.                             int y = this._bm.Height - (int)(yScale * (kv.Key.Y - yMin) + this.Margin.Bottom);
  205.                             this._g.DrawString(kv.Value, SystemFonts.DefaultFont, br, new Point(x, y));
  206.                         }
  207.                     }
  208.                 }
  209.             }while(false);
  210.         }

  211.         public Rectangle Margin
  212.         {
  213.             set;
  214.             get;
  215.         }

  216.         public Color BackgroundColor
  217.         {
  218.             set;
  219.             get;
  220.         }

  221.         public Color AxisColor
  222.         {
  223.             set;
  224.             get;
  225.         }
  226.         public double AxisWidth
  227.         {
  228.             set;
  229.             get;
  230.         }

  231.         public List<KeyValuePair<double, string>> XAxisLables
  232.         {
  233.             get;
  234.             private set;
  235.         }
  236.         public List<KeyValuePair<double, string>> YAxisLables
  237.         {
  238.             get;
  239.             private set;
  240.         }

  241.         public bool IsDrawHorizontalLine
  242.         {
  243.             set;
  244.             get;
  245.         }

  246.         #region  -- Collection functions --
  247.         public void Add(System.Windows.Point[] pts, Color cl, int width, DashStyle style)
  248.         {
  249.             if (pts != null)
  250.             {
  251.                 this._lines.Add(
  252.                     new KeyValuePair<System.Windows.Point[], object[]>(
  253.                         pts,
  254.                         new object[] { style, cl, width, }
  255.                     )
  256.                 );
  257.             }
  258.         }

  259.         public void Add(System.Windows.Point[] pts, Color cl, int width)    // Default style is Solid
  260.         {
  261.             this.Add(pts, cl, width, DashStyle.Solid);
  262.         }
  263.         public void Add(System.Windows.Point[] pts, Color cl)   // Default width is 1
  264.         {
  265.             this.Add(pts, cl, 1);
  266.         }
  267.         public void Add(System.Windows.Point[] pts)  // Default color is black
  268.         {
  269.             this.Add(pts, Color.Black);
  270.         }

  271.         public bool Remove(int idx)
  272.         {
  273.             bool rs = false;
  274.             if (idx >= 0 && this._lines.Count > idx)
  275.             {
  276.                 this._lines.RemoveAt(idx);
  277.                 rs = true;
  278.             }
  279.             return rs;
  280.         }

  281.         public void Clear()
  282.         {
  283.             this._lines.Clear();
  284.         }

  285.         public void AddLabel(double x, double y, string s){
  286.             this._Labels.Add(new KeyValuePair<System.Windows.Point, string>(new System.Windows.Point(x,y), s));
  287.         }
  288.         public void RemoveLable(int idx)
  289.         {
  290.             if (idx >= 0 && this._Labels.Count > idx)
  291.             {
  292.                 this._Labels.RemoveAt(idx);
  293.             }
  294.         }
  295.         public void ClearLabels()
  296.         {
  297.             this._Labels.Clear();
  298.         }

  299.         #endregion -- Collection functions --

  300.         #region -- Helper functions
  301.         public static explicit operator Bitmap(LinearGraph obj)
  302.         {
  303.             Bitmap rsBm = null;
  304.             if (obj != null)
  305.             {
  306.                 rsBm = obj._bm;
  307.             }
  308.             return rsBm;
  309.         }

  310.         public void Dispose()
  311.         {
  312.             this._g = null;
  313.             this._bm.Dispose();

  314.             //Pen p =null;
  315.             //p.DashStyle =System.Drawing.Drawing2D.DashStyle.Solid;
  316.         }
  317.         #endregion #region -- Helper functions

  318.         #region private member and defines
  319.         // -- System graphic objects --
  320.         private Graphics _g = null;
  321.         private Bitmap _bm = null;

  322.         // -- Lines --
  323.         private enum LineStyle:int
  324.         {
  325.             PenStyle =0,
  326.             LineColor =1,
  327.             LineWidth =2,
  328.         }
  329.         private List<KeyValuePair<System.Windows.Point[], object[]>> _lines =null;
  330.         private List<KeyValuePair<System.Windows.Point, string>> _Labels = new List<KeyValuePair<System.Windows.Point, string>>();

  331.         #endregion private member and defines
  332.     }
  333. }
复制代码





欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2