我的编程空间,编程开发者的网络收藏夹
学习永远不晚

WPF基于物理像素怎么绘制图形

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

WPF基于物理像素怎么绘制图形

本文小编为大家详细介绍“WPF基于物理像素怎么绘制图形”,内容详细,步骤清晰,细节处理妥当,希望这篇“WPF基于物理像素怎么绘制图形”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

从一个例子:
从FrameworkElement继承一个MyRectangleElement,然后重写OnRender方法如下:

protected override void OnRender(DrawingContext drawingContext){    Pen pen = new Pen(Brushes.Black, 1);    Rect rect = new Rect(20,20, 50, 60);     drawingContext.DrawRectangle(null, pen, rect);}

然后在Window上呈现MyRectangleElement,效果(右下角有放大镜)如下:

WPF基于物理像素怎么绘制图形

通过放大图形,能够很清晰地看到线条是不均匀的,在宏观视觉效果上感觉模糊,看上去很不舒服。于是,两个问题就产生了:

  • 为什么会出现这样的问题?

  • 如何解决?

怎么办?MSDN,百度,Google一起上!皇天不负苦心人,现在,谜底来到了我们的面前:

1、为什么会出现这样的效果呢?

WPF呈现引擎的反锯齿系统将那些没有在物理像素系统上的线扩展到了多个像素上。这里涉及到以下三个概念:

  • 物理像素系统:与物理图形设备相关。可以简单地理解为一个像素点的二维矩阵;

  • 逻辑像素系统:我们在画法方法中所使用的定位系统;

  • 反锯齿效果:一种计算机图形学上的算法,使得图形边缘更光滑;

声明:以上三个概念的解释是根据我个人的理解,不一定准确、严谨。如果您发现什么不对的地方,还望不吝指正。
为了更好地帮助您理解这个过程,这里给出一个示意图来解释一下:

WPF基于物理像素怎么绘制图形

上图中的淡蓝色网格可以视为“物理像素系统”,深色的图案是您画出的线条。可以很明显地看出,这条线的四条边都不在“物理像素系统”上,因此,“反锯齿系统”会将此线条的四条边扩展到相应的“物理像素系统”上。于是,本文最开始的情景便出现了。

2、如何解决这个问题?

针对不同的问题上下文,WPF给出了与之相应的解决方案。据我所知,有如下几个:

  • 1> 对于UIElement及其子类,使用SnapsToDevicePixels属性

对于从UIElement继承的类型,比如Control,通过将SnapsToDevicePixels设置为true可以得到清晰的图像,该属性的默认值为false。FrameworkElement从UIElement继承的时候,给这个属性赋予了一个Inherited元数据。如此一来,只要您在FrameworkElement Tree的根结点上将此属性设置为true,那么整个FrameworkElement Tree的绘制都将变得清晰起来。

  • 2> 对于自定义画法,使用GuidelineSet

SnapsToDevicePixels属性对于WPF Control来说是有用的,但是对本文的问题无能为力,于是,嗯,GuidelineSet横空出世!对于这个类,MSDN只是给出了一个用法的示例,从这个例子中我只能看到GuidelineSet可以这么用,但为什么是这样用就没有答案了。而且,有点离谱、神奇的是:MSDN上关于这点上一个示意图内容上有错。如下图所示:

WPF基于物理像素怎么绘制图形

图下部的左黑框是用了GuidelineSet后出现的结果,右边才是没有使用的结果。这两个图的位置应该互换一下。

我们必须回答这个问题:如果在(x1,y1) (x2,y2)处画一条线该如何在DrawingContext上使用GuidelineSet,以保证画法是清晰的呢?

这个问题让我着实纳闷了许久(原因本文第一段已经交代),不过,经过不断地尝试和思考,最终我找到了答案:
Guideline其实是图形设备在呈现时用来把逻辑像素点对齐到物理像素点的参考量。 使用它告诉图形设备你希望哪些逻辑像素点被对齐到物理像素点上。

声明:以上概念的解释是根据我个人的理解,不一定准确、严谨。如果您发现什么不对的地方,还望不吝指正。

下面,我将使用一个简单的示例来演示如何使用GuidelineSet,以及它所带来的效果。在这个示例中,我们使用DrawingContext的DrawLine方法绘制一个10×10的网格,相关代码如下:
首先,定义画法所用到的常量

    internal static class DrawingConstants    {        public static readonly int Rows = 10;        public static readonly int Columms = 10;        public static readonly double PenThickness = 1.0;        public static readonly double HalfOfPenThickness = PenThickness/2;    }

然后,定义NormalDrawingElement(使用一般画法):

    class NormalDrawingElement : FrameworkElement    {        protected override void OnRender(System.Windows.Media.DrawingContext drawingContext)        {            base.OnRender(drawingContext);            double xOffset = Math.Floor(this.RenderSize.Width / DrawingConstants.Columms);            double yOffset = Math.Floor(this.RenderSize.Height / DrawingConstants.Rows);            double xLineWidth = Math.Floor(this.RenderSize.Width);            double yLineHeight = Math.Floor(this.RenderSize.Height);            DrawingContext dct = drawingContext;            Pen blackPen = new Pen(Brushes.Black, DrawingConstants.PenThickness);            blackPen.Freeze();            //Draw the horizontal lines              Point x = new Point(0, 0);            Point y = new Point(xLineWidth, 0);            for (int i = 0; i <= DrawingConstants.Rows; i++)            {                dct.DrawLine(blackPen, x, y);                x.Offset(0, yOffset);                y.Offset(0, yOffset);            }            //Draw the vertical lines              x = new Point(0, 0);            y = new Point(0, yLineHeight);            for (int i = 0; i <= DrawingConstants.Columms; i++)            {                dct.DrawLine(blackPen, x, y);                x.Offset(xOffset, 0);                y.Offset(xOffset, 0);            }        }    }

定义GuidelineSetDrawingElement(使用GuidelineSet):

    class GuidelineSetDrawingElement : FrameworkElement    {        protected override void OnRender(System.Windows.Media.DrawingContext drawingContext)        {            base.OnRender(drawingContext);            double xOffset = Math.Floor(this.RenderSize.Width / DrawingConstants.Columms);            double yOffset = Math.Floor(this.RenderSize.Height / DrawingConstants.Rows);            double xLineWidth = Math.Floor(this.RenderSize.Width);            double yLineHeight = Math.Floor(this.RenderSize.Height);            DrawingContext dct = drawingContext;            Pen blackPen = new Pen(Brushes.Black, DrawingConstants.PenThickness);            blackPen.Freeze();            //Draw the horizontal lines              Point x = new Point(0, 0);            Point y = new Point(xLineWidth, 0);            for (int i = 0; i <= DrawingConstants.Rows; i++)            {                dct.PushGuidelineSet(new GuidelineSet(null, new double[] { y.Y - DrawingConstants.HalfOfPenThickness, y.Y + DrawingConstants.HalfOfPenThickness}));                dct.DrawLine(blackPen, x, y);                dct.Pop();                x.Offset(0, yOffset);                y.Offset(0, yOffset);            }            //Draw the vertical lines              x = new Point(0, 0);            y = new Point(0, yLineHeight);            for (int i = 0; i <= DrawingConstants.Columms; i++)            {                dct.PushGuidelineSet(new GuidelineSet(new double[] { x.X + DrawingConstants.HalfOfPenThickness, x.X - DrawingConstants.HalfOfPenThickness}, null));                dct.DrawLine(blackPen, x, y);                dct.Pop();                x.Offset(xOffset, 0);                y.Offset(xOffset, 0);            }        }    }

这个画法和上一个画法的区别仅仅在于以下两点:

  • 对于水平方向的线,我期望它的两个水平边缘是和”物理像素系统“对齐的;

  • 对于垂直方向的线,我期望它的两个垂直边缘是和”物理像素系统“对齐的。

最后,我们将这两个Element呈现出来:

<Window x:Class="GuidelineSetDemo.MainWindow"        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"        xmlns:loc="clr-namespace:GuideLineSetDemo"        Title="MainWindow" Height="350" Width="525">    <Grid>        <Grid.ColumnDefinitions>            <ColumnDefinition Width="*" />            <ColumnDefinition Width="*" />        </Grid.ColumnDefinitions>        <Grid Grid.Column="0">            <Grid.RowDefinitions>                <RowDefinition Height="Auto"/>                <RowDefinition Height="*"/>            </Grid.RowDefinitions>            <TextBlock Text="Normal Drawing" Margin="3" HorizontalAlignment="Center"/>            <loc:NormalDrawingElement Margin="10" Grid.Row="1"/>        </Grid>        <Grid Grid.Column="1">            <Grid.RowDefinitions>                <RowDefinition Height="Auto"/>                <RowDefinition Height="*"/>            </Grid.RowDefinitions>            <TextBlock Text="Dawing with GuidelineSet" Margin="3" HorizontalAlignment="Center"/>            <loc:GuidelineSetDrawingElement Margin="10" Grid.Row="1"/>        </Grid>    </Grid></Window>

运行后的效果如下:

WPF基于物理像素怎么绘制图形

另外,使用GuidelineSet的时候需要注意以下几个细节:

  • Push和pop一般情况下要成对(其实当您调用DrawingContext上的PushXXX方法后,都要考虑是否有与之对应的Pop方法调用);

  • GuidelineSet只对水平或者垂直线有用;

  • 使用GuidelineSet后,您所绘制图形的位置或大小可能和最初的设定有细微的差别。

读到这里,这篇“WPF基于物理像素怎么绘制图形”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

WPF基于物理像素怎么绘制图形

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

WPF基于物理像素怎么绘制图形

本文小编为大家详细介绍“WPF基于物理像素怎么绘制图形”,内容详细,步骤清晰,细节处理妥当,希望这篇“WPF基于物理像素怎么绘制图形”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。从一个例子:从FrameworkE
2023-07-02

基于Matlab怎么绘制小提琴图

今天小编给大家分享一下基于Matlab怎么绘制小提琴图的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。violinChart
2023-06-30

基于Qt OpenCV怎么实现图像灰度化像素

这篇文章主要介绍了基于Qt OpenCV怎么实现图像灰度化像素的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇基于Qt OpenCV怎么实现图像灰度化像素文章都会有所收获,下面我们一起来看看吧。效果图实现代码#i
2023-07-02

基于Python matplotlib库怎么绘制箱线图

这篇文章主要介绍了基于Python matplotlib库怎么绘制箱线图的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇基于Python matplotlib库怎么绘制箱线图文章都会有所收获,下面我们一起来看看吧
2023-06-30

基于Python+Matplotlib怎么实现直方图的绘制

今天小编给大家分享一下基于Python+Matplotlib怎么实现直方图的绘制的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧
2023-06-30

canvas里面怎么基于随机点绘制一个多边形

这篇文章主要介绍canvas里面怎么基于随机点绘制一个多边形,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!起因今天在学习《HTML5+Javascript动画基础》这本书的时候,在第八章的第三节讲到如何用三个弹簧连接
2023-06-09

Python怎么用PIL图像处理库绘制国际象棋棋盘

本篇内容介绍了“Python怎么用PIL图像处理库绘制国际象棋棋盘”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!目录1 PIL绘制国际象棋棋
2023-06-20

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录