扇形统计图

原文作者:ArcherSong

博客地址:https://www.cnblogs.com/ganbei/

  • 绘制一个扇形原理也是基于Canvas进行绘制;

  • ArcSegment[1]绘制弧形;

  • 绘制指示线;

  • 绘制文本;

  • 鼠标移入动画;

  • 显示详情Popup

  • 源码Github[2]Gitee[3]

1)SectorChart.cs代码如下;

using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Controls.Primitives;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Media.Effects;
using System.Windows.Shapes;
using WPFDevelopers.Charts.Models;namespace WPFDevelopers.Charts.Controls
{[TemplatePart(Name = CanvasTemplateName, Type = typeof(Canvas))][TemplatePart(Name = PopupTemplateName, Type = typeof(Popup))]public class SectorChart : Control{const string CanvasTemplateName = "PART_Canvas";const string PopupTemplateName = "PART_Popup";private Canvas _canvas;private Popup _popup;private double centenrX, centenrY, radius, offsetX, offsetY;private Point minPoint;private double fontsize = 12;private bool flg = false;public Brush Fill{get { return (Brush)GetValue(FillProperty); }set { SetValue(FillProperty, value); }}public static readonly DependencyProperty FillProperty =DependencyProperty.Register("Fill", typeof(Brush), typeof(SectorChart), new PropertyMetadata(null));public string Text{get { return (string)GetValue(TextProperty); }set { SetValue(TextProperty, value); }}public static readonly DependencyProperty TextProperty =DependencyProperty.Register("Text", typeof(string), typeof(SectorChart), new PropertyMetadata(null));public ObservableCollection<PieSerise> ItemsSource{get { return (ObservableCollection<PieSerise>)GetValue(ItemsSourceProperty); }set { SetValue(ItemsSourceProperty, value); }}public static readonly DependencyProperty ItemsSourceProperty =DependencyProperty.Register("ItemsSource", typeof(ObservableCollection<PieSerise>), typeof(SectorChart), new PropertyMetadata(null, new PropertyChangedCallback(ItemsSourceChanged)));private static void ItemsSourceChanged(DependencyObject d, DependencyPropertyChangedEventArgs e){var view = d as SectorChart;if (e.NewValue != null)view.DrawArc();}static SectorChart(){DefaultStyleKeyProperty.OverrideMetadata(typeof(SectorChart), new FrameworkPropertyMetadata(typeof(SectorChart)));}public override void OnApplyTemplate(){base.OnApplyTemplate();_canvas = GetTemplateChild(CanvasTemplateName) as Canvas;_popup = GetTemplateChild(PopupTemplateName) as Popup;}void DrawArc(){if (ItemsSource is null || !ItemsSource.Any() || _canvas is null)return;_canvas.Children.Clear();var pieWidth = _canvas.ActualWidth > _canvas.ActualHeight ? _canvas.ActualHeight : _canvas.ActualWidth;var pieHeight = _canvas.ActualWidth > _canvas.ActualHeight ? _canvas.ActualHeight : _canvas.ActualWidth;centenrX = pieWidth / 2;centenrY = pieHeight / 2;radius = this.ActualWidth > this.ActualHeight ? this.ActualHeight / 2 : this.ActualWidth / 2;double angle = 0;double prevAngle = 0;var sum = ItemsSource.Select(ser => ser.Percentage).Sum();foreach (var item in ItemsSource){var line1X = radius * Math.Cos(angle * Math.PI / 180) + centenrX;var line1Y = radius * Math.Sin(angle * Math.PI / 180) + centenrY;angle = item.Percentage / sum * 360 + prevAngle;double arcX = 0;double arcY = 0;if (ItemsSource.Count() == 1 && angle == 360){arcX = centenrX + Math.Cos(359.99999 * Math.PI / 180) * radius;arcY = (radius * Math.Sin(359.99999 * Math.PI / 180)) + centenrY;}else{arcX = centenrX + Math.Cos(angle * Math.PI / 180) * radius;arcY = (radius * Math.Sin(angle * Math.PI / 180)) + centenrY;}var line1Segment = new LineSegment(new Point(line1X, line1Y), false);bool isLargeArc = item.Percentage / sum > 0.5;var arcWidth = radius;var arcHeight = radius;var arcSegment = new ArcSegment();arcSegment.Size = new Size(arcWidth, arcHeight);arcSegment.Point = new Point(arcX, arcY);arcSegment.SweepDirection = SweepDirection.Clockwise;arcSegment.IsLargeArc = isLargeArc;var line2Segment = new LineSegment(new Point(centenrX, centenrY), false);PieBase piebase = new PieBase();piebase.Title = item.Title;piebase.Percentage = item.Percentage;piebase.PieColor = item.PieColor;piebase.LineSegmentStar = line1Segment;piebase.ArcSegment = arcSegment;piebase.LineSegmentEnd = line2Segment;piebase.Angle = item.Percentage / sum * 360;piebase.StarPoint = new Point(line1X, line1Y);piebase.EndPoint = new Point(arcX, arcY);var pathFigure = new PathFigure(new Point(centenrX, centenrY), new List<PathSegment>(){line1Segment,arcSegment,line2Segment,}, true);var pathFigures = new List<PathFigure>(){pathFigure,};var pathGeometry = new PathGeometry(pathFigures);var path = new Path() { Fill = item.PieColor, Data = pathGeometry, DataContext = piebase };_canvas.Children.Add(path);prevAngle = angle;var line3 = DrawLine(path);if (line3 != null)piebase.Line = line3;var textPathGeo = DrawText(path);var textpath = new Path() { Fill = item.PieColor, Data = textPathGeo };piebase.TextPath = textpath;_canvas.Children.Add(textpath);path.MouseMove += Path_MouseMove1;path.MouseLeave += Path_MouseLeave;if (ItemsSource.Count() == 1 && angle == 360){_canvas.Children.Add(line3);}else{var outline1 = new Line(){X1 = centenrX,Y1 = centenrY,X2 = line1Segment.Point.X,Y2 = line1Segment.Point.Y,Stroke = Brushes.White,StrokeThickness = 0.8,};var outline2 = new Line(){X1 = centenrX,Y1 = centenrY,X2 = arcSegment.Point.X,Y2 = arcSegment.Point.Y,Stroke = Brushes.White,StrokeThickness = 0.8,};_canvas.Children.Add(outline1);_canvas.Children.Add(outline2);_canvas.Children.Add(line3);}}}private void Path_MouseLeave(object sender, MouseEventArgs e){_popup.IsOpen = false;var path = sender as Path;var dt = path.DataContext as PieBase;TranslateTransform ttf = new TranslateTransform();ttf.X = 0;ttf.Y = 0;path.RenderTransform = ttf;dt.Line.RenderTransform = new TranslateTransform(){X = 0,Y = 0,};dt.TextPath.RenderTransform = new TranslateTransform(){X = 0,Y = 0,};path.Effect = new DropShadowEffect(){Color = (Color)ColorConverter.ConvertFromString("#FF949494"),BlurRadius = 20,Opacity = 0,ShadowDepth = 0};flg = false;}private void Path_MouseMove1(object sender, MouseEventArgs e){Path path = sender as Path;//动画if (!flg){BegionOffsetAnimation(path);}ShowMousePopup(path, e);}void ShowMousePopup(Path path, MouseEventArgs e){var data = path.DataContext as PieBase;if (!_popup.IsOpen)_popup.IsOpen = true;var mousePosition = e.GetPosition((UIElement)_canvas.Parent);_popup.HorizontalOffset = mousePosition.X + 20;_popup.VerticalOffset = mousePosition.Y + 20;Text = (data.Title + " : " + data.Percentage);//显示鼠标当前坐标点Fill = data.PieColor;}void BegionOffsetAnimation(Path path){NameScope.SetNameScope(this, new NameScope());var pathDataContext = path.DataContext as PieBase;var angle = pathDataContext.Angle;minPoint = new Point(Math.Round(pathDataContext.StarPoint.X + pathDataContext.EndPoint.X) / 2, Math.Round(pathDataContext.StarPoint.Y + pathDataContext.EndPoint.Y) / 2);var v1 = minPoint - new Point(centenrX, centenrY);var v2 = new Point(2000, 0) - new Point(0, 0);double vAngle = 0;if (180 < angle && angle <= 360 && pathDataContext.Percentage / ItemsSource.Select(p => p.Percentage).Sum() >= 0.5){vAngle = Math.Round(Vector.AngleBetween(v2, -v1));}else{vAngle = Math.Round(Vector.AngleBetween(v2, v1));}offsetX = 10 * Math.Cos(vAngle * Math.PI / 180);offsetY = 10 * Math.Sin(vAngle * Math.PI / 180);var line3 = pathDataContext.Line;var textPath = pathDataContext.TextPath;TranslateTransform LineAnimatedTranslateTransform =new TranslateTransform();this.RegisterName("LineAnimatedTranslateTransform", LineAnimatedTranslateTransform);line3.RenderTransform = LineAnimatedTranslateTransform;TranslateTransform animatedTranslateTransform =new TranslateTransform();this.RegisterName("AnimatedTranslateTransform", animatedTranslateTransform);path.RenderTransform = animatedTranslateTransform;TranslateTransform TextAnimatedTranslateTransform =new TranslateTransform();this.RegisterName("TextAnimatedTranslateTransform", animatedTranslateTransform);textPath.RenderTransform = animatedTranslateTransform;DoubleAnimation daX = new DoubleAnimation();Storyboard.SetTargetProperty(daX, new PropertyPath(TranslateTransform.XProperty));daX.Duration = new Duration(TimeSpan.FromSeconds(0.2));daX.From = 0;daX.To = offsetX;DoubleAnimation daY = new DoubleAnimation();Storyboard.SetTargetName(daY, nameof(animatedTranslateTransform));Storyboard.SetTargetProperty(daY, new PropertyPath(TranslateTransform.YProperty));daY.Duration = new Duration(TimeSpan.FromSeconds(0.2));daY.From = 0;daY.To = offsetY;path.Effect = new DropShadowEffect(){Color = (Color)ColorConverter.ConvertFromString("#2E2E2E"),BlurRadius = 33,Opacity = 0.6,ShadowDepth = 0};animatedTranslateTransform.BeginAnimation(TranslateTransform.XProperty, daX);animatedTranslateTransform.BeginAnimation(TranslateTransform.YProperty, daY);LineAnimatedTranslateTransform.BeginAnimation(TranslateTransform.XProperty, daX);LineAnimatedTranslateTransform.BeginAnimation(TranslateTransform.YProperty, daY);TextAnimatedTranslateTransform.BeginAnimation(TranslateTransform.XProperty, daX);TextAnimatedTranslateTransform.BeginAnimation(TranslateTransform.YProperty, daY);flg = true;}/// <summary>/// 画指示线/// </summary>/// <param name="path"></param>/// <returns></returns>Polyline DrawLine(Path path){NameScope.SetNameScope(this, new NameScope());var pathDataContext = path.DataContext as PieBase;var angle = pathDataContext.Angle;pathDataContext.Line = null;minPoint = new Point(Math.Round(pathDataContext.StarPoint.X + pathDataContext.EndPoint.X) / 2, Math.Round(pathDataContext.StarPoint.Y + pathDataContext.EndPoint.Y) / 2);Vector v1;if (angle > 180 && angle < 360){v1 = new Point(centenrX, centenrY) - minPoint;}else if (angle == 180 || angle == 360){if (Math.Round(pathDataContext.StarPoint.X) == Math.Round(pathDataContext.EndPoint.X)){v1 = new Point(radius * 2, radius) - new Point(centenrX, centenrY);}else{if (Math.Round(pathDataContext.StarPoint.X) - Math.Round(pathDataContext.EndPoint.X) == 2 * radius){v1 = new Point(radius, 2 * radius) - new Point(centenrX, centenrY);}else{v1 = new Point(radius, 0) - new Point(centenrX, centenrY);}}}else{v1 = minPoint - new Point(centenrX, centenrY);}v1.Normalize();var Vmin = v1 * radius;var RadiusToNodal = Vmin + new Point(centenrX, centenrY);var v2 = new Point(2000, 0) - new Point(0, 0);double vAngle = 0;vAngle = Math.Round(Vector.AngleBetween(v2, v1));offsetX = 10 * Math.Cos(vAngle * Math.PI / 180);offsetY = 10 * Math.Sin(vAngle * Math.PI / 180);var prolongPoint = new Point(RadiusToNodal.X + offsetX * 1, RadiusToNodal.Y + offsetY * 1);if (RadiusToNodal.X == double.NaN || RadiusToNodal.Y == double.NaN || prolongPoint.X == double.NaN || prolongPoint.Y == double.NaN)return null;var point1 = RadiusToNodal;var point2 = prolongPoint;Point point3;if (prolongPoint.X >= radius)point3 = new Point(prolongPoint.X + 10, prolongPoint.Y);elsepoint3 = new Point(prolongPoint.X - 10, prolongPoint.Y);PointCollection polygonPoints = new PointCollection();polygonPoints.Add(point1);polygonPoints.Add(point2);polygonPoints.Add(point3);var line3 = new Polyline();line3.Points = polygonPoints;line3.Stroke = pathDataContext.PieColor;pathDataContext.PolylineEndPoint = point3;return line3;}PathGeometry DrawText(Path path){NameScope.SetNameScope(this, new NameScope());var pathDataContext = path.DataContext as PieBase;Typeface typeface = new Typeface(new FontFamily("Microsoft YaHei"),FontStyles.Normal,FontWeights.Normal, FontStretches.Normal);FormattedText text = new FormattedText(pathDataContext.Title,new System.Globalization.CultureInfo("zh-cn"),FlowDirection.LeftToRight, typeface, fontsize, Brushes.RosyBrown);var textWidth = text.Width;Geometry geo = null;if (pathDataContext.PolylineEndPoint.X > radius)geo = text.BuildGeometry(new Point(pathDataContext.PolylineEndPoint.X + 4, pathDataContext.PolylineEndPoint.Y - fontsize / 1.8));elsegeo = text.BuildGeometry(new Point(pathDataContext.PolylineEndPoint.X - textWidth - 4, pathDataContext.PolylineEndPoint.Y - fontsize / 1.8));PathGeometry pathGeometry = geo.GetFlattenedPathGeometry();return pathGeometry;}}
}

2)SectorChart.xaml 代码如下;

<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:controls="clr-namespace:WPFDevelopers.Charts.Controls"><Style TargetType="{x:Type controls:SectorChart}"><Setter Property="Width" Value="300"/><Setter Property="Height" Value="300"/><Setter Property="Template"><Setter.Value><ControlTemplate TargetType="{x:Type controls:SectorChart}"><Grid><Popup x:Name="PART_Popup" IsOpen="False"Placement="Relative" AllowsTransparency="True"><Border Background="White" CornerRadius="5" Padding="14"BorderThickness="0"BorderBrush="Transparent"><StackPanel ><Ellipse Width="20" Height="20"Fill="{TemplateBinding Fill}"/><TextBlock Background="White" Padding="9,4,9,4" TextWrapping="Wrap" Text="{TemplateBinding Text}"/></StackPanel></Border></Popup><Canvas x:Name="PART_Canvas"  HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}"Width="{TemplateBinding ActualWidth}"Height="{TemplateBinding ActualHeight}"></Canvas></Grid></ControlTemplate></Setter.Value></Setter></Style>
</ResourceDictionary>

3) MainWindow.xaml使用如下;

xmlns:wsCharts="https://github.com/WPFDevelopersOrg.WPFDevelopers.Charts"<wsCharts:SectorChart  ItemsSource="{Binding ItemsSource,RelativeSource={RelativeSource AncestorType=local:MainWindow}}"Margin="30" />

4) MainWindow.xaml.cs代码如下;

using System.Collections.ObjectModel;
using System.Windows;
using System.Windows.Media;
using WPFDevelopers.Charts.Models;namespace WPFDevelopers.Charts.Samples
{/// <summary>/// MainWindow.xaml 的交互逻辑/// </summary>public partial class MainWindow {public ObservableCollection<PieSerise> ItemsSource{get { return (ObservableCollection<PieSerise>)GetValue(ItemsSourceProperty); }set { SetValue(ItemsSourceProperty, value); }}public static readonly DependencyProperty ItemsSourceProperty =DependencyProperty.Register("ItemsSource", typeof(ObservableCollection<PieSerise>), typeof(MainWindow), new PropertyMetadata(null));public MainWindow(){InitializeComponent();Loaded += MainWindow_Loaded;}private void MainWindow_Loaded(object sender, RoutedEventArgs e){ItemsSource = new ObservableCollection<PieSerise>();var collection1 = new ObservableCollection<PieSerise>();collection1.Add(new PieSerise{Title = "2012",Percentage = 30,PieColor = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#5B9BD5")),});collection1.Add(new PieSerise{Title = "2013",Percentage = 140,PieColor = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#4472C4")),});collection1.Add(new PieSerise{Title = "2014",Percentage = 49,PieColor = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#007fff")),});collection1.Add(new PieSerise{Title = "2015",Percentage = 50,PieColor = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#ED7D31")),});collection1.Add(new PieSerise{Title = "2016",Percentage = 30,PieColor = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#FFC000")),});collection1.Add(new PieSerise{Title = "2017",Percentage = 30,PieColor = new SolidColorBrush((Color)ColorConverter.ConvertFromString("#ff033e")),});ItemsSource = collection1;}}
}

参考资料

[1]

ArcSegment: https://docs.microsoft.com/zh-cn/dotnet/api/system.windows.media.arcsegment?view=windowsdesktop-6.0

[2]

Github: https://github.com/WPFDevelopersOrg/WPFDevelopers.Charts

[3]

Gitee: https://gitee.com/WPFDevelopersOrg/WPFDevelopers.Charts

WPF 实现扇形统计图相关推荐

  1. php扇形分布图,PHP制作3D扇形统计图以及对图片进行缩放操作实例

    这篇文章主要介绍了PHP制作3D扇形统计图以及对图片进行缩放操作实例,需要的朋友可以参考下 1.利用php gd库的函数绘制3D扇形统计图 header("content-type" ...

  2. Asp.Net实例:C# 绘制统计图(三) ——扇形统计图的绘制

    三. 扇形统计图的绘制<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" /& ...

  3. WPF实现特殊统计图

    WPF实现特殊统计图 原文:WPF实现特殊统计图 效果图: ActiveFunItem.xaml代码: <UserControl x:Class="SunCreate.Vipf.Cli ...

  4. 优秀课程案例|如何用scratch画扇形统计图

    如何用scratch画扇形统计图 一.学习目标 1.有六个班,每班人数是50-100间的随机数,用扇形统计图统计出各班人数所占的比例图(各班人数列表假设是一个集合,集合的性质决定元素互异性,每个元素不 ...

  5. php扇形统计图怎么做,PHP实现绘制3D扇形统计图及图片缩放实例_php实例

    1.利用php gd库的函数绘制3D扇形统计图 50; $i--) { imagefilledarc($image, 50, $i, 100, 50, -160, 40, $darknavy, IMG ...

  6. iOS 实现扇形统计图

    今天项目增加新功能,某页面需要扇形统计图,宝宝拿来分享一下~ 前面两个.h .m文件是某位大神已经写好的.后面是使用方法. 需要遵守DataSource和Delegate协议. XYPieChart. ...

  7. 基于Winform的Chart控件的简单使用(Chart控件中的条形统计图、折线统计图、扇形统计图的简单使用)

    Chart控件集成了颇多的统计图模型,拿来即用的理念大大节省了开发的时间.下面演示最常见的三种统计图模型的使用. 效果展示: C#代码: using System; using System.Coll ...

  8. C# Winfrm窗体 chart控件应用(条形统计图,折线统计图,扇形统计图)

    **首先**,统计图的重要作用,小编在这里就不多说了,而我们的C#做统计图在chart控件的加入下显得格外轻松,下面开始正题. **接下来**,创建**CreateImage**窗体,拖入两个char ...

  9. html js 扇形统计图_canvas绘制扇形统计图

    今天偷懒了,把上次的作业代码记录在这里好了. 绘制简单的扇形统计图需要用到JS插件,绘制完以后的代码是这样的: 今天有点晚了把代码记录一下: *{ margin:0px; padding:0px;} ...

最新文章

  1. NESPER的大体结构 z
  2. python课程将主要介绍哪些内容-Python窗口的基本介绍
  3. java jdbc 教程_java JDBC系列教程之JDBC类的简析与JDBC的基础操作
  4. python语言设计简单计算器_Python 设计一个简单的计算器-Go语言中文社区
  5. Wannafly挑战赛22 D 整数序列 (线段树维护三角函数值)
  6. FinTech领域的风险控制——策略篇
  7. android sdk引入 微信分享_Android 微信SDK分享功能中的最全过程步骤分析
  8. 转: hibernate配置文件hibernate.cfg.xml和.hbm.xml的详细解释
  9. 940mx黑苹果驱动_黑苹果intel网卡驱动方法
  10. 面试官:说说什么是 Java 内存模型(JMM)?
  11. 对程序员而言,学历重要吗?
  12. 谷歌学术首页url爬取
  13. 打卡day01 python基础—常用数据类型
  14. 传感器系列之4.10 酒精传感器
  15. 罗技 连点 脚本_双十一,罗技,价格不贵又好用的办公无线鼠标,吐血推荐!2020.10月,双11...
  16. ubuntu12.04编译安装crtmpserver
  17. 微信婚恋相亲交友小程序
  18. 仿真工具ABAQUS
  19. 黑苹果MacOS Monterey 12.0 -12.3 OpenCore 0.8.0 (OC) 引导升级教程(最新版)
  20. 用于工作的 Linux 桌面——Ubuntu 22.04

热门文章

  1. Oracle使用——PLSQL的中文乱码显示全是问号
  2. Micro Motion™(高准)ELITE™ 科里奥利流量和密度仪表
  3. 微信小程序开发学习—Day2
  4. linux rwx 权限
  5. Linux ❉ 权限位rwx
  6. wordcloud生成词云图(含形状、颜色设置)
  7. 春节前感冒了,体会到了健康的重要性
  8. 解决国内访问不了github的问题
  9. 寝室打扫卫生负责人查询
  10. 【引用】引用的概念与基本使用原则