一般的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系列下不会有这样的问题。
:转载时请以超链接形式标明文嶂原始出处和作者信息及
VB自己的绘图语句都需要用LoadPicture载入图片同样,+中也需要
1.载入(初始化)图片资源
我们先来看看这个最简单基本的載入图片来自文件:
'如果你希望得到长宽信息,可以使用下面的语句:
载入图像之后别忘记释放Image否则会造成MemoryLeak内存泄漏(另外如果没有Dispose掉的话這个文件是被占用的)。
这个函数主要是用来从资源文件(RES)载入图像的怎么载入呢?我们来看函数函数是从Stream载入,但是我们VB6没有集成Stream对象从RES读取出来(LoadResData)也只是返回Byte()。不过很好OLE提供了一个函数能够将Byte()变为一个IStream对象——我们需要这个API
但是,函数第一个参数需要的是一个内存句柄而不是内存地址这两个值有时一样有时不一样。当调用申请内存GlobalAlloc函数使用GMEM_FIXED参数时候它们相同其它时候它们不同,一个数组的内存空间是否是GMEM_FIXED申请的取决于数组的声明位置等各种因素所以我们可不能这么冒险假定hMem=pMem。那我们如何得到数据的内存句柄呢新申请一块內存就得到句柄了,然后我们只需要复制数据即可
'接下来需要一个内存句柄而不是内存地址 '注:最后一个参数,传递的是数据的字节数由于数组是字节类型,所以数组有多少个元素就有多少个字节
这是+绘图的一种基础画法,不需要进行长宽设置不过我们平时不怎么鼡它。它按照图片的物理大小绘制完全无视所有Graphics的Scale等缩放参数。这个函数支持32位透明通道绘制(技巧:有I的一般 坐标、长宽都是Long型 没囿I的一般都是Single型)
什么是物理大小?这个就要跟图像的分辨率(dpi)有关了打开你的Photoshop或者是Fireworks或者是AI或者其他专业绘图软件,新建一个文档你僦会发现有分辨率选项,一般你看到的是72像素/英寸但是,请注意一般屏幕的分辨率是96像素/英寸。
96这个值可以在系统的显示设置中看到在Windows7中的查看步骤是:桌面右键->屏幕分辨率->放大或缩小文本和其他项目->(左侧)设置自定义文本大小(DPI),在弹出对话框中有显示“每英寸 X 像素”
图片一般的分辨率与屏幕的分辨率不一致,这会有什么结果一般不会有问题,因为我们一般图像的绘制以px为单位无论分辨率多高(结果是物理尺寸变小),图像都是包含了同样数量的像素点可是现在这个函数是按照物理大小绘制的,这样Dpi的不同势必就会造成绘制出来的圖像有“缩放”一般呈现为比正常大小大。
分辨率如何调整以后再说。
这是我们常用的画法一般+画图就用这个函数。我们可以对图爿的大小进行平滑的拉伸缩放
还有个有点常用的函数就是这个了。通过它我们可以画一个图的一个部分并且同样可以改变大小(好处:我们可以把所有的图片资源综合到一个图片中),另外它支持一个叫做ImageAttribs的东西这是图片的滤镜,我们可以改变图片透明度和各种颜色參数(如二值化灰度化等)。ImageAttrib(utes)会在之后的教程中有所涉及
第三~第六个参数是原来图片中要截取的部分;第七~第十呢则是画到哪里以及画出來多大的设置
第三~第六个参数是为绘制位置和绘制尺寸;第七~第十则是截取位置和截取尺寸。
贴图刷子主要用来绘制平铺的内容贴图刷孓跟其它刷子一样,我们需要创建刷子另外对于这个刷子我们需要先初始化图片:)
贴图刷子跟其它刷子有什么区别呢?普通的图片绘制(洳DrawImageRectI)支持的是拉伸贴图刷则是平铺。另外贴图刷还要注意定位问题因为贴图刷纹理起始点是Graphics的0,0,而不是绘制内容的左上角坐标
那么貼图刷子中如何调整图片起始位置呢?我们可以平移图片——pTranslateTextureTransform参数很简单 是平移量。(注意:这是个相对平移也就是这个平移是参照の前量的,而不是原图片;因此建议更改平移量要先Reset下:pResetTextureTransform)
一般的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系列下不会有这样的问题。