WPF 玫瑰图绘制

Posted funk

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了WPF 玫瑰图绘制相关的知识,希望对你有一定的参考价值。

1.前言:

一直在从事CS应用程序开发工作,随着工作需求,要对部分数据进行可视化展示,UI设计稿其中就有玫瑰图、雷达图的展示。

花了一个下午回溯原来丢掉的数学知识点。。特此将实现方法记录下。

2.效果图:

技术图片

 

 

 

 3.数据对象(RadarObj)

  每个图都是由一个数据集合对象组成,从而绘制出对应的效果,对象最基本的属性要有某一维度的数值,用于在图像中展示。

 

    public class RadarObj
    
        public string RColor  get; set; 
        public string Name  get; set; 
        public int DataValue  get; set; 
        public double DataRaidus  get; set; 

        /// <summary>
        /// Series stroke
        /// </summary>
        public Brush Stroke
        
            get
            
                return new SolidColorBrush((Color)ColorConverter.ConvertFromString(RColor)); ;
            
        

        /// <summary>
        /// Series Fill
        /// </summary>
        public Brush Fill
        
            get
            
                return new SolidColorBrush((Color)ColorConverter.ConvertFromString(RColor));
            
        
    

  

 4.绘制玫瑰图

 核心逻辑在于,将玫瑰图先理解为一个饼图,然后根据数值计算出在饼图中占用的角度,以及对应的扇面半径,改动每个扇面的半径就成了玫瑰图

 其中需要使用到几何的一些基本概念,工作这么多年,忘记了蛮多的,后面各种恶补。直接上代码吧。

<UserControl x:Class="Painter.NightingaleRose"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:local="clr-namespace:Painter"
             mc:Ignorable="d" 
             d:DesignHeight="450" d:DesignWidth="800">
    <Grid>
        <Canvas x:Name="CanvasPanel" HorizontalAlignment="Center" VerticalAlignment="Center" Background="Gray" >
        </Canvas>
    </Grid>
</UserControl>
 public partial class NightingaleRose : UserControl
    
        public NightingaleRose()
        
            InitializeComponent();
        

        #region Property

        /// <summary>
        /// 数据
        /// </summary>
        public List<RadarObj> Datas
        
            get  return (List<RadarObj>)GetValue(DatasProperty); 
            set  SetValue(DatasProperty, value); 
        

        /// <summary>
        /// 数值的总数
        /// </summary>
        public int Count
        
            get
            
                return Datas.Sum(i => i.DataValue);
            
        

        public static readonly DependencyProperty DatasProperty = DependencyProperty.Register("Datas", typeof(List<RadarObj>),
        typeof(NightingaleRose), new PropertyMetadata(new List<RadarObj>()));

        /// <summary>
        /// 当前绘制大区域
        /// </summary>
        private double MaxSize
        
            get
            
                var par = this.Parent as FrameworkElement;
                return par.ActualHeight > par.ActualWidth ? par.ActualWidth : par.ActualHeight;
            
        

        /// <summary>
        /// 停靠间距
        /// </summary>
        public int RoseMargin
        
            get  return (int)GetValue(RoseMarginProperty); 
            set  SetValue(RoseMarginProperty, value); 
        

        public static readonly DependencyProperty RoseMarginProperty = DependencyProperty.Register("RoseMargin", typeof(int),
        typeof(NightingaleRose), new PropertyMetadata(50));

        /// <summary>
        /// 空心内环半径
        /// </summary>
        public int RoseInsideMargin
        
            get  return (int)GetValue(RoseInsideMarginProperty); 
            set  SetValue(RoseInsideMarginProperty, value); 
        

        public static readonly DependencyProperty RoseInsideMarginProperty = DependencyProperty.Register("RoseInsideMargin", typeof(int),
        typeof(NightingaleRose), new PropertyMetadata(20));

        /// <summary>
        /// 显示值标注
        /// </summary>
        public bool ShowValuesLabel
        
            get  return (bool)GetValue(ShowValuesLabelProperty); 
            set  SetValue(ShowValuesLabelProperty, value); 
        

        public static readonly DependencyProperty ShowValuesLabelProperty = DependencyProperty.Register("ShowValuesLabel", typeof(bool),
        typeof(NightingaleRose), new PropertyMetadata(true));


        public static readonly DependencyProperty ShowToolTipProperty = DependencyProperty.Register("ShowToolTip", typeof(bool),
        typeof(NightingaleRose), new PropertyMetadata(false));

        /// <summary>
        /// 延伸线长
        /// </summary>
        public int LabelPathLength
        
            get  return (int)GetValue(LabelPathLengthProperty); 
            set  SetValue(LabelPathLengthProperty, value); 
        

        public static readonly DependencyProperty LabelPathLengthProperty = DependencyProperty.Register("LabelPathLength", typeof(int),
        typeof(NightingaleRose), new PropertyMetadata(50));

        #endregion Property

        #region Method

        /// <summary>
        /// 初始化数据
        /// </summary>
        private void initData()
        
            CanvasPanel.Children.Clear();
            if (this.Datas != null && this.Datas.Count > 0)
            
                this.CanvasPanel.Width = this.CanvasPanel.Height = 0;

                //求角度比例尺  (每个值占多大的角度  可以算到每一块图所占的角度)
                var angelScale = 360.00 / Datas.Sum(i => i.DataValue);

                //最大半径
                var maxRadius = (MaxSize / 2) - RoseMargin - (ShowValuesLabel ? LabelPathLength : 0);

                //半径比例尺  (值和比例尺相乘等于每一块图的半径)
                var radiusScale = maxRadius / Datas.Max(o => o.DataValue);

                //计算半径宽度值
                for (int i = 0; i < Datas.Count; i++)
                
                    Datas[i].DataRaidus = Datas[i].DataValue * radiusScale;
                
                //扇形角度初始化
                double angleSectorStart = 0;
                double angleSectorEnd = 0;

                //循环绘制扇形区域
                int scaleTimeSpan = 0;
                int pathTimespan = 0;
                int textTimeSpan = 0;
                for (int index = 0; index < Datas.Count; index++)
                
                    //计算扇形角度
                    if (index == 0)
                    
                        angleSectorStart = 0;
                        angleSectorEnd = Datas[index].DataValue * angelScale;
                    
                    else if (index + 1 == Datas.Count)
                    
                        angleSectorStart += Datas[index - 1].DataValue * angelScale;
                        angleSectorEnd = 360;
                    
                    else
                    
                        angleSectorStart += Datas[index - 1].DataValue * angelScale;
                        angleSectorEnd = angleSectorStart + Datas[index].DataValue * angelScale;
                    
                    var currentRadius = RoseInsideMargin + Datas[index].DataRaidus;
                    //计算扇形点位,用于绘制PATH
                    Point ptOutSideStart = GetPoint(currentRadius, angleSectorStart * Math.PI / 180);
                    Point ptOutSideEnd = GetPoint(currentRadius, angleSectorEnd * Math.PI / 180);
                    Point ptInSideStart = GetPoint(RoseInsideMargin, angleSectorStart * Math.PI / 180);
                    Point ptInSideEnd = GetPoint(RoseInsideMargin, angleSectorEnd * Math.PI / 180);
                    if (string.IsNullOrEmpty(Datas[index].RColor) )
                        Datas[index].RColor = ChartColorPool.ColorStrings[index];
                    Path pthSector = new Path()  Fill = Datas[index].Fill ;
                    //PATH数据格式 M0,100  L50,100 A50,50 0 0 1 100,50 L100,0 A100,100 0 0 0 0,100 Z
                    StringBuilder datastrb = new StringBuilder();

                    #region BuilderPathData

                    datastrb.Append("M");
                    datastrb.Append(ptOutSideStart.X.ToString());
                    datastrb.Append(",");
                    datastrb.Append(ptOutSideStart.Y.ToString());
                    datastrb.Append(" L");
                    datastrb.Append(ptInSideStart.X.ToString());
                    datastrb.Append(",");
                    datastrb.Append(ptInSideStart.Y.ToString());
                    datastrb.Append(" A");
                    datastrb.Append(RoseInsideMargin.ToString());
                    datastrb.Append(",");
                    datastrb.Append(RoseInsideMargin.ToString());
                    datastrb.Append(" 0 0 1 ");
                    datastrb.Append(ptInSideEnd.X.ToString());
                    datastrb.Append(",");
                    datastrb.Append(ptInSideEnd.Y.ToString());
                    datastrb.Append(" L");
                    datastrb.Append(ptOutSideEnd.X.ToString());
                    datastrb.Append(",");
                    datastrb.Append(ptOutSideEnd.Y.ToString());
                    datastrb.Append(" A");
                    datastrb.Append(currentRadius.ToString());
                    datastrb.Append(",");
                    datastrb.Append(currentRadius.ToString());
                    datastrb.Append(" 0 0 0 ");
                    datastrb.Append(ptOutSideStart.X.ToString());
                    datastrb.Append(",");
                    datastrb.Append(ptOutSideStart.Y.ToString());
                    datastrb.Append(" Z");

                    #endregion BuilderPathData
                    try
                    
                        pthSector.Data = (Geometry)new GeometryConverter().ConvertFromString(datastrb.ToString());
                    
                    catch (Exception exp)
                     
                    //设置扇形显示的动画
                    AnimationUtils.FloatElement(pthSector,1, 200, pathTimespan += 200);
                    AnimationUtils.ScaleRotateEasingAnimationShow(pthSector,0.1,1,1500, scaleTimeSpan += 200,null );

                    CanvasPanel.Children.Add(pthSector);
                    if (ShowValuesLabel)
                    
                        //计算延伸线角度
                        double lbPathAngle = angleSectorStart + (angleSectorEnd - angleSectorStart) / 2;
                        //起点
                        Point ptLbStart = GetPoint(currentRadius, lbPathAngle * Math.PI / 180);
                        //终点
                        Point ptLbEnd = GetPoint(maxRadius + LabelPathLength, lbPathAngle * Math.PI / 180);
                        Path pthLb = new Path()  Stroke = Datas[index].Stroke, StrokeThickness = 1 ;
                        pthLb.Data = (Geometry)new GeometryConverter().ConvertFromString(string.Format("M0,1 2,3", ptLbStart.X.ToString(), ptLbStart.Y.ToString(), ptLbEnd.X.ToString(), ptLbEnd.Y.ToString()));
                        double dur = (textTimeSpan += 200) + 1500;
                        AnimationUtils.CtrlDoubleAnimation(pthLb, 1000, dur);
                        CanvasPanel.Children.Add(pthLb);
                        SetLabel(Datas[index], ptLbEnd, dur);
                    
                
                this.SizeChanged -= RadarControl_SizeChanged;
                this.SizeChanged += RadarControl_SizeChanged;
            
        

        public void InitalControl()
        
        

        /// <summary>
        /// 初始化数据
        /// </summary>
        /// <param name="dataobj"></param>
        public void SetData(object dataobj)
        
            this.Datas = (dataobj) as List<RadarObj>;
            this.initData();
        

        private void RadarControl_SizeChanged(object sender, SizeChangedEventArgs e)
        
            initData();
        

        #endregion Method

        #region Compare

        /// <summary>
        /// 计算点位
        /// </summary>
        /// <param name="radius"></param>
        /// <param name="angel"></param>
        /// <returns></returns>
        private Point GetPoint(double radius, double angel)
        
            return new Point(radius * Math.Cos(angel), radius * Math.Sin(angel));
        

        private void SetLabel(RadarObj obj, Point location,double duration)
        
            //计算偏移量
            bool x = true;
            bool y = true;

            if (location.X < 0)
                x = false;
            if (location.Y < 0)
                y = false;
            //obj.Name + " " + 
            TextBlock txb = new TextBlock()  Text = " "+obj.Name+" "+Getbfb(Count.ToString(), obj.DataValue.ToString(), 2)+ " ", Foreground = this.Foreground, FontSize = this.FontSize ;
            Size s = ControlSizeUtils.GetTextAreaSize(txb.Text, this.FontSize);
            CanvasPanel.Children.Add(txb);
            AnimationUtils.CtrlDoubleAnimation(txb, 1000, duration);
            if (location.X > -5 && location.X < 5)
                Canvas.SetLeft(txb, location.X - (s.Width / 2));
            else
                Canvas.SetLeft(txb, location.X + (x ? 0 : -(s.Width)));

            if (location.Y > -5 && location.Y < 5)
                Canvas.SetTop(txb, location.Y - (s.Height / 2));
            else
                Canvas.SetTop(txb, location.Y + (y ? 0 : -(s.Height)));
        
        /// <summary>
        /// 计算百分比
        /// </summary>
        /// <param name="zs">总数</param>
        /// <param name="tj">当前项的值</param>
        /// <param name="num">保留的小数点几位</param>
        /// <returns></returns>
        public static string Getbfb(string zs, string tj, int num)
        
            try
            
                if (zs.Equals("0"))
                
                    return "0";
                
                double bfb = (double.Parse(tj) / double.Parse(zs)) * 100;
                if (bfb >= 100)
                
                    bfb = 100;
                

                return Math.Round(bfb, num).ToString() + "%";
            
            catch (Exception ex)
            
                return "0%";
            
        

        #endregion Compare

       
    

 

调用示例

 在主窗体grid里面丢个按钮,按钮点击之后执行下列代码

 

private void RoseClick(object sender, RoutedEventArgs e)
        
            NightingaleRose rdc = new NightingaleRose();
            this.GrdMain.Children.Clear();
            this.GrdMain.Children.Add(rdc);
            rdc.SetData(CrData());
        
        private List<RadarObj> CrData()
        
            List<RadarObj> list = new List<RadarObj>();
            list.Add(new RadarObj()  Name="A", DataValue= rdm.Next(20,100) );
            list.Add(new RadarObj()  Name = "B", DataValue = rdm.Next(20, 100) );
            list.Add(new RadarObj()  Name = "C", DataValue = rdm.Next(20, 100) );
            list.Add(new RadarObj()  Name = "D", DataValue = rdm.Next(20, 100) );
            list.Add(new RadarObj()  Name = "E", DataValue = rdm.Next(20, 100) );
            list.Add(new RadarObj()  Name = "F", DataValue = rdm.Next(20, 100) );
            list.Add(new RadarObj()  Name = "F", DataValue = rdm.Next(20, 100) );
            return list;
        

  

以上是关于WPF 玫瑰图绘制的主要内容,如果未能解决你的问题,请参考以下文章

python3使用matplotlib绘制风速风向玫瑰图

数据可视化应用绘制风玫瑰图(附Python代码)

WPF 雷达图

MATLAB表白玫瑰花绘制——旋转玫瑰蓝色玫瑰

python表白玫瑰花绘制——情人节表白

python表白玫瑰花绘制——情人节表白