使用 WPF 和 C# 绘制图形

绘图困难

在 WPF 和 C# 中绘制图形

此示例展示了如何在 WPF 和 C# 中绘制图形。绘制图形总是很棘手,因为您通常需要在至少两个不同的坐标系中工作。首先,您要为图形使用世界坐标。例如,您可能希望 X 值的范围为 2000 年至 2020 年,Y 值的范围为 10,000 美元至 100,000 美元之间的销售额值。

第二个坐标系是以屏幕上的像素为单位测量的 设备坐标系。

显然,在绘制图形本身之类的东西时,您需要使用世界坐标系。最棘手的部分发生在您需要在世界坐标中定位某些东西但在设备坐标中绘制时。例如,假设您要绘制带有 5 个像素长的刻度标记的 X 轴和 Y 轴。您使用世界坐标来确定刻度标记应放置在何处,但随后您需要计算设备坐标中刻度标记的长度(以像素为单位)。

类似地,假设您想在图表上绘制一些文本来标记某些内容。您将文本定位在世界坐标中,但您可能希望在设备坐标中绘制文本。否则很难将文本居中并对齐。

我要提到的最后一个怪异问题是让图形的线条具有一致的粗细。假设您在某个规范化的空间中绘制图形,然后缩放它以适合设备区域。例如,您在世界坐标空间 2000 <= x <= 2020、$10,000 <= y <= $100,000 中绘制图形,然后使用 LayoutTransform使图形适合Canvas控件。当变换拉伸图形时。它还会拉伸您为图形绘制的线条。除非垂直和水平比例因子相同,否则线条在垂直和水平方向上的拉伸量会不同。文本也会被拉伸,从而产生一些非常烦人的结果。

无论如何,要真正将所有东西都准确地放置在您想要的位置,您需要能够在世界坐标和设备坐标中自由工作。这篇文章是一系列简短文章的开篇,这些文章探讨了您可以用来在 WPF 和 C# 中绘制图形的技术。

一个简单的图表

此示例仅使用设备坐标绘制了一个简单的图形。换句话说,所有位置都以像素为单位测量,左上角为 (0, 0)。以下帖子将展示如何在更方便的世界坐标中工作。

以下代码显示了构建该程序的 XAML。

<Window x:Class="howto_wpf_graph.Window1"xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"Title="howto_wpf_graph"Height="250" Width="335" Loaded="Window_Loaded"><Grid Background="LightGreen"><Canvas Name="canGraph" Background="White"Width="300" Height="200"VerticalAlignment="Center" HorizontalAlignment="Center"/></Grid>
</Window>

该窗口的主要子窗口是Grid,其中包含一个名为canGraph的Canvas

在 WPF 中,您通常不会直接在绘图表面上绘图。如果确实需要,您可以这样做,但通常使用LineEllipseRectangle和其他形状控件进行绘制。如果愿意,您可以将这些对象包含在 XAML 代码中,但如果您要绘制非平凡图形,则需要使用代码来完成。

当此示例启动时,以下事件处理程序将构建图形。(请注意窗口的 XAML 声明中的Loaded="Window_Loaded"部分。这告诉程序Window_Loaded方法是窗口的Loaded事件的事件处理程序。)

// Draw a simple graph.
private void Window_Loaded(object sender, RoutedEventArgs e)
{const double margin = 10;double xmin = margin;double xmax = canGraph.Width - margin;double ymin = margin;double ymax = canGraph.Height - margin;const double step = 10;// Make the X axis.GeometryGroup xaxis_geom = new GeometryGroup();xaxis_geom.Children.Add(new LineGeometry(new Point(0, ymax), new Point(canGraph.Width, ymax)));for (double x = xmin + step;x <= canGraph.Width - step; x += step){xaxis_geom.Children.Add(new LineGeometry(new Point(x, ymax - margin / 2),new Point(x, ymax + margin / 2)));}Path xaxis_path = new Path();xaxis_path.StrokeThickness = 1;xaxis_path.Stroke = Brushes.Black;xaxis_path.Data = xaxis_geom;canGraph.Children.Add(xaxis_path);// Make the Y ayis.GeometryGroup yaxis_geom = new GeometryGroup();yaxis_geom.Children.Add(new LineGeometry(new Point(xmin, 0), new Point(xmin, canGraph.Height)));for (double y = step; y <= canGraph.Height - step; y += step){yaxis_geom.Children.Add(new LineGeometry(new Point(xmin - margin / 2, y),new Point(xmin + margin / 2, y)));}Path yaxis_path = new Path();yaxis_path.StrokeThickness = 1;yaxis_path.Stroke = Brushes.Black;yaxis_path.Data = yaxis_geom;canGraph.Children.Add(yaxis_path);// Make some data sets.Brush[] brushes = { Brushes.Red, Brushes.Green, Brushes.Blue };Random rand = new Random();for (int data_set = 0; data_set < 3; data_set++){int last_y = rand.Next((int)ymin, (int)ymax);PointCollection points = new PointCollection();for (double x = xmin; x <= xmax; x += step){last_y = rand.Next(last_y - 10, last_y + 10);if (last_y < ymin) last_y = (int)ymin;if (last_y > ymax) last_y = (int)ymax;points.Add(new Point(x, last_y));}Polyline polyline = new Polyline();polyline.StrokeThickness = 1;polyline.Stroke = brushes[data_set];polyline.Points = points;canGraph.Children.Add(polyline);}
}

代码首先为图定义一些边界。

接下来,程序创建一个GeometryGroup对象来表示 X 轴。GeometryGroup可以容纳其他几何对象,例如线条。代码创建一个Line表示轴的基线并将其添加到组中。然后,它使用循环创建一组Line对象来表示刻度标记并将它们添加到组中。

在创建完所有轴的Line对象并将其添加到GeometryGroup后,程序将创建一个Path对象并设置其StrokeThicknessStroke属性。然后它将路径的Data属性设置为等于GeometryGroup

最后,代码将路径添加到canGraph对象的Children集合中。

然后代码重复这些步骤来创建 Y 轴。

接下来,代码生成一些图形数据。对于每个数据集,代码都会创建一个PointCollection对象。它会生成一堆随机点并将它们添加到集合中。完成数据生成后,程序会创建一个Polyline,设置其绘制属性,并将其Points属性设置为点集合。最后,代码将Polyline添加到canGraph对象的Children集合中。

这就是程序需要做的全部工作。当窗口出现时,LinePolyline对象会根据需要自行绘制。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/766.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

3D滤波器处理遥感tif图像

import cv2 import numpy as np from osgeo import gdal# 定义 Gabor 滤波器的参数 kSize 31 # 滤波器核的大小 g_sigma 3.0 # 高斯包络的标准差 g_theta np.pi / 4 # Gabor 函数的方向 g_lambda 10.0 # 正弦波的波长 g_gamma 0.5 # 空间纵横比 g_psi np.pi / 2 # …

JuiceFS 2024:开源与商业并进,迈向 AI 原生时代

即将过去的 2024 年&#xff0c;是 JuiceFS 开源版本推出的第 4 年&#xff0c;企业版的第 8 个年头。回顾过去这一年&#xff0c;JuiceFS 社区版依旧保持着快速成长的势头&#xff0c;GitHub 星标突破 11.1K&#xff0c;各项使用指标增长均超过 100%&#xff0c;其中文件系统总…

高可用虚拟IP-keepalived

个人觉得华为云这个文档十分详细&#xff1a;使用虚拟IP和Keepalived搭建高可用Web集群_弹性云服务器 ECS_华为云 应用场景&#xff1a;虚拟IP技术。虚拟IP&#xff0c;就是一个未分配给真实主机的IP&#xff0c;也就是说对外提供数据库服务器的主机除了有一个真实IP外还有一个…

支付宝租赁小程序提升租赁行业效率与用户体验

内容概要 在当今数字化的世界里&#xff0c;支付宝租赁小程序的出现构建了一种新的租赁模式&#xff0c;使得用户在使用过程中体验更加流畅。想象一下&#xff0c;你在寻找租赁服务时&#xff0c;不再需要繁琐的流程和冗长的等待&#xff0c;只需通过手机轻松点击几下&#xf…

python异常机制

异常是什么&#xff1f; 软件程序在运行过程中&#xff0c;非常可能遇到刚刚提到的这些问题&#xff0c;我们称之为异常&#xff0c;英文是Exception&#xff0c;意思是例外。遇到这些例外情况&#xff0c;或者交异常&#xff0c;我们怎么让写的程序做出合理的处理&#xff0c…

Git撤销指定commit并更新远端仓库

Git撤销指定commit并更新远端仓库 一、撤销指定commit 1.首先执行git log 命令&#xff0c;查看git历史提交以及commit信息&#xff1a; 由于需要脱敏&#xff0c;所以截图可能看得马赛克比较多&#xff0c;需要关注的就是上面的commit后跟的id&#xff0c;以及HEAD当前指定…

【opencv】第8章 图像轮廓与图像分割修复

8.1 查找并绘制轮廓 一个轮廓一般对应一系列的点&#xff0c;也就是图像中的一条曲线。其表示方法可能 根据不同的情况而有所不同。在OpenCV 中&#xff0c;可以用findContours()函数从二值图 像中查找轮廓 8.1.1 寻找轮廓&#xff1a; findContours() 函数 findContours) 函…

.NET Core NPOI 导出图片到Excel指定单元格并自适应宽度

NPOI&#xff1a;支持xlsx&#xff0c;.xls&#xff0c;版本>2.5.3 XLS&#xff1a;HSSFWorkbook&#xff0c;主要前缀HSS&#xff0c; XLSX&#xff1a;XSSFWorkbook&#xff0c;主要前缀XSS&#xff0c;using NPOI.XSSF.UserModel; 1、导出Excel添加图片效果&#xff0…

SQL分类与数据类型整理

SQL分类与数据类型整理 SQL分类数据类型数值型整数型小数型布尔型字符串型日期型二进制型其他类型&#xff08;空间数据类型&#xff09; 总结 SQL&#xff0c;全称为Structured Query Language&#xff08;结构化查询语言&#xff09;&#xff0c;是一种高度专业化的计算机语言…

基于深度学习算法的AI图像视觉检测

基于人工智能和深度学习方法的现代计算机视觉技术在过去10年里取得了显著进展。如今&#xff0c;它被广泛用于图像分类、人脸识别、图像中物体的识别等。那么什么是深度学习&#xff1f;深度学习是如何应用在视觉检测上的呢&#xff1f; 什么是深度学习&#xff1f; 深度学习是…

30_Redis哨兵模式

在Redis主从复制模式中,因为系统不具备自动恢复的功能,所以当主服务器(master)宕机后,需要手动把一台从服务器(slave)切换为主服务器。在这个过程中,不仅需要人为干预,而且还会造成一段时间内服务器处于不可用状态,同时数据安全性也得不到保障,因此主从模式的可用性…

计算机网络 笔记 网络层1

网络层功能概述 主要的任务是把分组从源端传输到目的端&#xff0c;为分组交换网上的不同主句提供通信服务&#xff0c;网络层的传输单位是数据报。 主要的功能&#xff1b; 1&#xff0c;路由选择&#xff1a;路由选择指网络层根据特定算法&#xff0c;为数据包从源节点到目…

解决计算机管理无法连接远程电脑

症状 有时&#xff0c;我们会想用计算机管理&#xff08;Computer Management&#xff09;连接到一台远程Windows服务器&#xff0c;因为我们不想直接登录到远程服务器上操作。 然而&#xff0c;绝大多数情况下&#xff0c;如果你初次尝试连接&#xff0c;会得到如下的错误。 …

【Uniapp-Vue3】Prop校验与prop默认值用法及循环遍历数组对象

一、prop校验 如果我们在想要限制prop的类型&#xff0c;就可以在接收prop的时候对接收类型进行限制&#xff1a; defineProps({ 属性名:{ type:类型 } }) 需要注意类型的首字母大写 但是设置了传入参数类型限制并不能严格限制&#xff0c;只会在后台进行提示&#xff1a; 二、…

解决WordPress出现Fatal error: Uncaught TypeError: ftp_nlist()致命问题

错误背景 WordPress版本&#xff1a;wordpress-6.6.2-zh_CN WooCommerce版本&#xff1a;woocommerce.9.5.1 WordPress在安装了WooCommerce插件后&#xff0c;安装的过程中没有问题&#xff0c;在安装完成后提示&#xff1a; 此站点遇到了致命错误&#xff0c;请查看您站点管理…

qt-C++笔记之自定义继承类初始化时涉及到parents的初始化

qt-C笔记之自定义继承类初始化时涉及到parents的初始化 code review! 参考笔记 1.qt-C笔记之父类窗口、父类控件、对象树的关系 2.qt-C笔记之继承自 QWidget和继承自QObject 并通过 getWidget() 显示窗口或控件时的区别和原理 3.qt-C笔记之自定义类继承自 QObject 与 QWidget …

ElasticSearch | Elasticsearch与Kibana页面查询语句实践

关注&#xff1a;CodingTechWork 引言 在当今大数据应用中&#xff0c;Elasticsearch&#xff08;简称 ES&#xff09;以其高效的全文检索、分布式处理能力和灵活的查询语法&#xff0c;广泛应用于各类日志分析、用户行为分析以及实时数据查询等场景。通过 ES&#xff0c;用户…

Java聊天小程序

拟设计一个基于 Java 技术的局域网在线聊天系统,实现客户端与服务器之间的实时通信。系统分为客户端和服务器端两类,客户端用于发送和接收消息,服务器端负责接收客户端请求并处理消息。客户端通过图形界面提供用户友好的操作界面,服务器端监听多个客户端的连接并管理消息通…

【Redis学习 | 第5篇】Redis缓存 —— 缓存的概念 + 缓存穿透 + 缓存雪崩 + 缓存击穿

文章目录 完成任务1. 什么是缓存2. 添加商户缓存3. 缓存更新策略3.1 主动更新 4. 缓存穿透5. 缓存雪崩6. 缓存击穿6.1 使用互斥锁查询商铺信息6.2 使用逻辑过期查询商铺信息 7. 封装 Redis 工具类 完成任务 1. 什么是缓存 缓存&#xff1a;数据交换的缓冲区&#xff08;Cache…

单元测试概述入门

引入 什么是测试&#xff1f;测试的阶段划分&#xff1f; 测试方法有哪些&#xff1f; 1.什么是单元测试&#xff1f; 单元测试&#xff1a;就是针对最小的功能单元&#xff08;方法&#xff09;&#xff0c;编写测试代码对其正确性进行测试。 2.为什么要引入单元测试&#x…