c#繪制儀表

jklynm 10年前發布 | 1K 次閱讀 C# 繪制,儀表

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace GMapTest
{
    public partial class FrmMeter : Form
    {
        public FrmMeter()
        {
            InitializeComponent();
            MaxValue = 270;
            ChangeValue = 69;
            DrawBackImg();
            //this.BackgroundImage = GMapTest.Properties.Resources.bdi;
        }

        //為了方式繪制指針時產生的閃爍,PictureBox添加該事件方法

        private void pic_Paint(object sender, PaintEventArgs e)
        {
            //DrawForeImg(e.Graphics);
        }


        private double _maxValue;
        public double MaxValue {
            get { return _maxValue; }
            set { _maxValue = value; }
        }

        private double _changeValue;
        public double ChangeValue
        {
            get { return _changeValue; }
            set
            {
                if (value <= _maxValue)
                    _changeValue = value;
                else
                {
                    //完成自適應性
                    MaxValue = value;
                    _changeValue = value;
                }
                //通過該方法,可以使指針自動繪制(其實就是強制重繪)

                //pic.Invalidate();
            }
        }

        //指針的具體畫法 

        private void DrawForeImg(Graphics gp)
        {
            Bitmap bit = new Bitmap(this.Width, this.Height);
            Graphics g = Graphics.FromImage(bit);
            g.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;

            //畫針
            DrawPin(g);
            DrawString(g);

            //注意此處的繪制方式,這樣可以有效減少界面的閃爍問題。
            gp.DrawImage(bit, new Point(0, 0));
            g.Dispose();

        }
        private readonly int _diameter = 300;

        private Color _pinColor = Color.CornflowerBlue;

        private string _unitStr = "AUC";

        private Color _frameColor = Color.SkyBlue;

        private StringFormat strFormat = StringFormat.GenericTypographic;

        private double _PinLen = 130;

        private double NxPinLen =10;

        //畫針
        private void DrawPin(Graphics g)
        {
            int cer = _diameter / 2;
            float start = 135;
            float sweepShot = (float)(_changeValue / _maxValue * 270);

            Pen linePen = new Pen(_pinColor, 1);
            Pen NxPen = new Pen(_pinColor, 2);
            Pen xPen = new Pen(_pinColor, 5);
            double rad = (sweepShot + start) * Math.PI / 180;
            float radius = _diameter / 2 - 5;
            int dx = (int)(cer + (_PinLen) * Math.Cos(rad));
            int dy = (int)(cer + (_PinLen) * Math.Sin(rad));

            int px = (int)(cer + (_PinLen * 0.4) * Math.Cos(rad));
            int py = (int)(cer + (_PinLen * 0.4) * Math.Sin(rad));

            int nx = (int)(cer - (NxPinLen) * Math.Sin(rad));
            int ny = (int)(cer - (NxPinLen) * Math.Cos(rad));
            g.DrawLine(linePen, new Point(cer, cer), new Point(dx, dy));
            g.DrawLine(NxPen, new Point(cer, cer), new Point(px, py));
            g.DrawLine(xPen, new Point(cer, cer), new Point(ny, nx));
        }

        //繪制在儀表下面的值

        private void DrawString(Graphics g)
        {
            int cer = _diameter / 2;
            string str = _changeValue.ToString("F2");
            g.DrawString(str, new Font("宋體", 9), new SolidBrush(_pinColor), new PointF(cer, (float)(cer + cer * 0.4)), strFormat);
        }

        // 繪制背景 用來總體控制背景的繪制
        private void DrawBackImg()
        {
            Bitmap bit = new Bitmap(this.Width, this.Height);
            Graphics gp = Graphics.FromImage(bit);
            gp.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;
            #region 在這里可以擴展需要繪制的背景項目
           
            //外框
            drawFrame(gp);
            // 畫刻度
            DrawRuling(gp);
            //畫點
            drawPoint(gp);

            //繪制單位
            DrawUnitStr(gp);

            #endregion

            //當繪制完成后,直接直接設置為背景
            DrawForeImg(gp);
            this.panel1.BackgroundImage = bit;
           
        }

        //繪制單位
        private void DrawUnitStr(Graphics gp)
        {
            int cer = _diameter / 2;
            gp.DrawString(_unitStr, new Font("宋體", 10), new SolidBrush(_frameColor), new PointF(cer, (float)(cer - cer * 0.3)), strFormat);

        }

        /// <summary>
        /// 畫外框
        /// </summary>
        /// <param name="gp"></param>
        private void drawFrame(Graphics gp)
        {
            Pen pen = new Pen(_frameColor, 2);
            Rectangle rec = new Rectangle(5, 5, _diameter - 10, _diameter - 10);
            Rectangle reci = new Rectangle(7, 7, _diameter - 14, _diameter - 14);
            gp.FillEllipse(Brushes.Black, rec);
            gp.DrawEllipse(pen, reci);
        }
        // 畫刻度  此次較為復雜,主要是在繪制刻度值時需要處理
        private void DrawRuling(Graphics gp)
        {
            //刻度
            int cerX = _diameter / 2;
            int cerY = _diameter / 2;

            //這里需要注意,因外在上面的圖中標識了rad=0的位置,而我們的儀表時270度的,0點在135度處,

            //為了符合該效果所以起始位置設為135度。
            float start = 135;
            float sweepShot = 0;
            int dx = 0;
            int dy = 0;
            int soildLenght = 8;
            Pen linePen = new Pen(Color.White, 1);
            float span = (float)(_maxValue / 30);
            float sp = 0;
            //用于右邊數字右對齊
            StringFormat stf = new StringFormat();
            stf.Alignment = StringAlignment.Far;

            StringFormat stfMid = new StringFormat();
            stfMid.Alignment = StringAlignment.Center;
            stfMid.LineAlignment = StringAlignment.Center;
            for (int i = 0; i <= 30; i++)
            {
                //注意此處,C#提供的三角函數計算中使用的弧度值,而此處獲取的是角度值,需要轉化

                double rad = (sweepShot + start) * Math.PI / 180;
                float radius = _diameter / 2 - 5;
                int px = (int)(cerX + (radius-20) * Math.Cos(rad));
                int py = (int)(cerY + (radius - 20) * Math.Sin(rad));
                if (sweepShot % 15 == 0)
                {
                    linePen.Width = 2;

                    //計算刻度中的粗線
                    dx = (int)(cerX + (radius - 20 - soildLenght) * Math.Cos(rad));
                    dy = (int)(cerY + (radius - 20 - soildLenght) * Math.Sin(rad));

                    //繪制刻度值,注意字串對其方式
                    string str = sp.ToString("f0");
                    if (sweepShot <= 45)
                    {
                        gp.DrawString(str, new Font("宋體", 9), new SolidBrush(_frameColor), new PointF(dx, dy - 5));
                    }
                    else if (sweepShot > 45 && sweepShot < 135)
                    {
                        gp.DrawString(str, new Font("宋體", 9), new SolidBrush(_frameColor), new PointF(dx, dy));
                    }
                    else if (sweepShot == 135)
                    {
                        gp.DrawString(str, new Font("宋體", 9), new SolidBrush(_frameColor), new PointF(dx, dy + 10), stfMid);
                    }
                    else if (sweepShot > 135 && sweepShot < 225)
                    {
                        gp.DrawString(str, new Font("宋體", 9), new SolidBrush(_frameColor), new PointF(dx, dy), stf);
                    }
                    else if (sweepShot >= 225)
                    {
                        gp.DrawString(str, new Font("宋體", 9), new SolidBrush(_frameColor), new PointF(dx, dy - 5), stf);
                    }

                }
                else
                {

                    //計算刻度中細線

                    linePen.Width = 1;
                    dx = (int)(cerX + (radius - 20 - soildLenght + 2) * Math.Cos(rad));
                    dy = (int)(cerY + (radius - 20 - soildLenght + 2) * Math.Sin(rad));
                }

                //繪制刻度線
                gp.DrawLine(linePen, new Point(px, py), new Point(dx, dy));
                sp += span;
                sweepShot += 9;
            }
        }
        //畫中間的點
        private void drawPoint(Graphics gp)
        {
            Pen p = new Pen(_frameColor);
            int tmpWidth = 6;
            int px = _diameter / 2 - tmpWidth;

            gp.DrawEllipse(p, new Rectangle(px, px, 2 * tmpWidth, 2 * tmpWidth));

            //在畫點時,我使用了指針的顏色,這樣看起來,更真實一點
            gp.FillEllipse(new SolidBrush(_pinColor), new Rectangle(px + 2, px + 2, 2 * tmpWidth - 4, 2 * tmpWidth - 4));
        }

        private void panel1_Paint(object sender, PaintEventArgs e)
        {
            //DrawBackImg();
        }      
    }

 


}

 本文由用戶 jklynm 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!