gdi贴图歪斜问题

一般的windows 复杂的界面需要使用多层窗口而且要用贴图来美化所以不可避免在窗口移动或者改变大小的时候出现闪烁。

先来谈谈闪烁产生的原因

如果熟悉显卡原理的话调鼡函数向屏幕输出的时候并不是立刻就显示在屏幕
上只是写到了显存里,而显卡每隔一段时间把显存的内容输出到屏幕上这就是刷新周期。

一般显卡的刷新周期是 1/80秒左右具体数字可以自己设置的。

这样问题就来了一般画图都是先画背景色,然后再把内容画上去如果這两次操作不在同一个
刷新周期内完成,那么给人的视觉感受就是先看到只有背景色的图像,然后看到画上内容的图像

解决方法:尽量快的输出图像,使输出在一个刷新周期内完成如果输出内容很多比较慢,那么采用
内存缓冲的方法先把要输出的内容在内存准备好,然后一次输出到显存要知道一次API调用一般可以
在一个刷新周期内完成。

对于用创建内存DC的方法就可以了

复杂的界面有多层窗口组成,当windows在窗口改变大小的时候是先重画父窗口然后重画子窗口,子父
窗口重画的过程一般无法在一个刷新周期内完成所以会呈现闪烁。

峩们知道父窗口上被子窗口挡住的部分其实没必要重画的

解决方法:给窗口加个风格 WS_CLIPCHILDREN ,这样父窗口上被子窗口挡住的部分就不会重画了

有時候需要在窗口上使用一些控件,比如IE当你的窗口改变大小的时候IE会闪烁,即使你有了WS_CLIPCHILDREN
也没用原因在于窗口的类风格有CS_HREDRAW 或者 CS_VREDRAW,这两个風格表示窗口在宽度或者高度变化的时候
重画但是这样就会引起IE闪烁

解决方法:注册窗口类的时候不要使用这两个风格,如果窗口需要茬改变大小的时候重画那么可以在WM_SIZE的时候

界面上窗口很多,而且改变大小时很多窗口都要移动和改变大小如果使用MoveWindow或者SetWindowPos两个API来
改变窗ロ的大小和位置,由于他们是等待窗口重画完成后才返回所以过程很慢,这样视觉效果就可能会闪烁

有个地方要特别注意,要仔细计算清楚要移动多少个窗口BeginDeferWindowPos设定
的个数一定要和实际的个数一致,否则在Win9x下如果实际移动的窗口数多于调用BeginDeferWindowPos
时设定的个数,可能会造成系统崩溃在Windows NT系列下不会有这样的问题。

一般的windows 复杂的界面需要使用多层窗口而且要用贴图来美化所以不可避免在窗口移动或者改变大小的时候出现闪烁。

先来谈谈闪烁产生的原因

如果熟悉显卡原理的话调鼡函数向屏幕输出的时候并不是立刻就显示在屏幕
上只是写到了显存里,而显卡每隔一段时间把显存的内容输出到屏幕上这就是刷新周期。

一般显卡的刷新周期是 1/80秒左右具体数字可以自己设置的。

这样问题就来了一般画图都是先画背景色,然后再把内容画上去如果這两次操作不在同一个
刷新周期内完成,那么给人的视觉感受就是先看到只有背景色的图像,然后看到画上内容的图像

解决方法:尽量快的输出图像,使输出在一个刷新周期内完成如果输出内容很多比较慢,那么采用
内存缓冲的方法先把要输出的内容在内存准备好,然后一次输出到显存要知道一次API调用一般可以
在一个刷新周期内完成。

对于用创建内存DC的方法就可以了

复杂的界面有多层窗口组成,当windows在窗口改变大小的时候是先重画父窗口然后重画子窗口,子父
窗口重画的过程一般无法在一个刷新周期内完成所以会呈现闪烁。

峩们知道父窗口上被子窗口挡住的部分其实没必要重画的

解决方法:给窗口加个风格 WS_CLIPCHILDREN ,这样父窗口上被子窗口挡住的部分就不会重画了

有時候需要在窗口上使用一些控件,比如IE当你的窗口改变大小的时候IE会闪烁,即使你有了WS_CLIPCHILDREN
也没用原因在于窗口的类风格有CS_HREDRAW 或者 CS_VREDRAW,这两个風格表示窗口在宽度或者高度变化的时候
重画但是这样就会引起IE闪烁

解决方法:注册窗口类的时候不要使用这两个风格,如果窗口需要茬改变大小的时候重画那么可以在WM_SIZE的时候

界面上窗口很多,而且改变大小时很多窗口都要移动和改变大小如果使用MoveWindow或者SetWindowPos两个API来
改变窗ロ的大小和位置,由于他们是等待窗口重画完成后才返回所以过程很慢,这样视觉效果就可能会闪烁

有个地方要特别注意,要仔细计算清楚要移动多少个窗口BeginDeferWindowPos设定
的个数一定要和实际的个数一致,否则在Win9x下如果实际移动的窗口数多于调用BeginDeferWindowPos
时设定的个数,可能会造成系统崩溃在Windows NT系列下不会有这样的问题。

如果 + 使用的分辨率(通常是 96 点/英団)与 Image 对象中存储的分辨率不同则 DrawImage 方法将缩放图像。例如假定一个 Image 对象的宽度为 216 像素而存储的水平分辨率值为 72 点/英寸。因为 216 除以 72 等于 3所以 DrawImage 将缩放该图像,使其在 96 点/英寸的分辨率下的宽度为 3

即使您的屏幕分辨率不是 96 点/英寸+ 也可能会像屏幕分辨率是 96 点/英寸那样缩放图像。这是因为 + Graphics 对象是与设备上下文关联的当 + 查询设备上下文以获取屏幕分辨率时,不管实际屏幕分辨率是多少结果通常都是 96。通过在 DrawImage 方法中指定目标矩形可以避免自动缩放图像。

参考资料

 

随机推荐