如何避免多控件窗体重新布局时闪烁

来源:互联网 发布:帝国cms整合ck 编辑:程序博客网 时间:2024/04/30 09:43

适用场景: 

需要在某容器控件中动态装载多个子控件,而且该容器控件可能需要改变WindowFormState,即从Normal转变为Maxmized,或者是其他状态转换啦, what ever :)

如果没有应用任何特殊处理,你就会发现,当容器控件状态转换时,其上的子控件在经过一阵狂闪之后(可能背景控件颜色和自身相互交替出现),最终恢复至平静; 这种情形当然无论是程序员自己和客户都不愿意看到的

ok,废话一通之后,咱们开始解释原因,以及考虑解决方案

为什么会闪烁?

因为窗体控件状态转换时,windows需要负责"擦除"其背景,重新绘制,在一台性能并不优良的终端上(很大可能程度上客户端电脑都不是那么强劲吧) ,这个过程不是一时半会就能完成的,尤其对于很多个子控件的情况,因此就…

解决之道?

如果稍微写过WinForm程序的同学,肯定或多或少的用过ListView控件,简单易用嘛 :) 那么一定也知道该控件有2个比较有意思的方法:

BeginUpdate

Prevents the control from drawing until the EndUpdate method is called.

EndUpdate

Resumes drawing of the list view control after drawing is suspended by the BeginUpdate method.

从msdn的解释来看,这2个方法的应用能解决往ListView控件中分多次Add ListViewItem时闪烁的问题,ok,既然它能这么处理,咱们自己的容器控件为什么不能依葫芦画瓢呢?

btw. 其实我一开始也没任何好方法解决闪烁问题,后来偶尔想到ListView的此特性 :)

看看ListView.BeginUpdateInternal方法怎么写:


internal void BeginUpdateInternal()
{
    
if (this.IsHandleCreated)
    {
        
if (this.updateCount == 0)
        {
            
this.SendMessage(1100);
        }
        
this.updateCount = (short) (this.updateCount + 1);
    }
}

关键一行在 this.SendMessage(11, 0, 0); 虾米意思呢? 它给自身Send了一个code为11的windows消息,11代表虾米?

在windows消息定义中可以看到 WM_SETREDRAW = 0x0B (0x0B也就是11),这行代码的意思是告诉windows对ListView控件停止重绘界面,直到显式要求重新绘制为止. 很牛叉对不对 :D
ok,在EndUpdateInternal中又做了虾米?

 


internal bool EndUpdateInternal(bool invalidate)
{
    
if (this.updateCount <= 0)
    {
        
return false;
    }
    
this.updateCount = (short) (this.updateCount - 1);
    
if (this.updateCount == 0)
    {
        
this.SendMessage(11-10);
        
if (invalidate)
        {
            
this.Invalidate();
        }
    }
    
return true;
}

 

同样有一行代码: this.SendMessage(11, –1, 0); 11还是同一个意思,此时告知windows可以重绘ListView控件了

ok,到这时候应该明白这2个方法含义了吧,也就是说对子控件的操作都是在一个“冻结”的状态中进行的,等到所有准备工作就绪,才对最终状态重新绘制,因此界面就不会出现闪烁状态.

如何依葫芦画瓢?

要知道不是每个控件类都有提供BeginUpdate和EndUpdate方法,所以需要自己亲自打造一个

1. SendMessage如何来? 从windows api interop而来,很简单,有个工具可以提供所有api函数到c#方法的转换: P/Invoke Interop Assistant

2.11这个定义以及类似的东东怎么找? 强大的google或者bing可以帮忙 :D

至于代码怎么写,就不需要偶来操刀啦 :D

 

后记:

这样处理之后,是不是发现闪烁从此就不再出现了?但是……还有问题

拿一个无边框窗体举例,当它从Normal状态变为Maxmized(顺便设置TopMost为true),你会很高兴看到该窗体包含的子控件真的不闪了,从Maxmized回到Normal时,也不闪了,但是很诡异的问题发生了: 任务栏不见了?取而代之的是当前窗体的背景色??? 难道任务栏没有重绘回来???

发生什么事情了? 不是只让窗体的重绘停止了吗,怎么会影响到任务栏窗口?

ok,解决办法是有的,发个消息给任务栏窗口让它强制重绘,怎么写?

同样,对于.net中的任何控件,都自带Invalidate方法,通过调用该方法,可以强制重绘控件的整体或某矩形部分, 又要画瓢啦

在Invalidate中可以看到这么一行

SafeNativeMethods.RedrawWindow(new HandleRef(this.window, this.Handle), (NativeMethods.COMRECT) null, NativeMethods.NullHandleRef, 0x85);

现在的问题是,如何获取任务栏窗口句柄?  bing一把就会发现,很简单: FindWindow(“Shell_TrayWnd”, “”);

ok, 后面的事情就简单啦

 

再后记

不难发现,Control对象其实是自带BeginUpdateInternal方法的,但m$很恶毒的把它弄成internal的了… 而且只被少数几个控件享用: ListView, ComboBox, DataGrid, TreeView

前面说到模仿Control类自身的BeginUpdateInternal和EndUpdateInternal方法,封装各个控件都适用的基础类,先看代码吧

复制代码
 1    class AvoidControlFlicker
 2    {
 3        private int _paintFrozen;
 4
 5        public void FreezePainting(Control toFreezeControl, bool isToFreeze)
 6        {
 7            if (null == toFreezeControl)
 8                throw new ArgumentNullException("toFreezeControl");
 9
10            if (isToFreeze && toFreezeControl.IsHandleCreated && toFreezeControl.Visible)
11            {
12                if (0 == _paintFrozen++)
13                {
14                    NativeMethods.SendMessage(toFreezeControl.Handle, NativeConsts.WM_SETREDRAW, 00);
15                }

16            }

17            if (!isToFreeze)
18            {
19                if (0 == _paintFrozen) return;
20                if (0 == --_paintFrozen)
21                {
22                    NativeMethods.SendMessage(toFreezeControl.Handle, NativeConsts.WM_SETREDRAW, 10);
23                    toFreezeControl.Invalidate(true);
24                }

25            }

26        }

27    }
复制代码
代码很简单, 当需要"冻结" 控件对象时,由windows向该控件发送WM_SETREDRAW事件, 不需要"冻结"时,将该事件再发一次,改变参数即可,注意的是
1.添加一个计数器,防止多次发同一个消息
2.将控件"解冻"后,需要强制让控件重绘,即23行代码: toFreezeControl.Invalidate(true); 因为Invalidate方法是Control类自身提供的,所以可以直接拿来用了,其中true参数表示该控件的所有子控件也一并重绘了,false表示只重绘自身

那么,如果针对外部窗体怎么写呢? (不属于该进程的窗体,或者无法直接访问的窗体对象),拿windows任务栏来说,需要先获取它的句柄:
IntPtr taskBarHandle = NativeMethods.FindWindowA("Shell_TrayWnd""");
拿到句柄后就可以给它发windows消息了:
NativeMethods.SendMessage(taskBarHandle, NativeConsts.WM_SETREDRAW, 00); //禁止重绘
需要"解冻"时:
NativeMethods.SendMessage(taskBarHandle, NativeConsts.WM_SETREDRAW, 10);
NativeMethods.RedrawWindow(taskBarHandle, IntPtr.Zero, IntPtr.Zero, NativeConsts.WM_NCPAINT); 
//强制重绘

至于demo代码就很简单了, 在窗体的resize事件我创建了225个Button对象,如果不应用AvoidControlFlicker类,可以很明显看出窗体缩放时有大块的空白区域,实际效果在demo压缩包里有2个截屏视频比较,废话不多说, 点我下载 

ps. 根据我测试的结果, 在vista上如果对任务栏禁止重绘,而没有启用重绘则后果很悲剧:鼠标点击之后根本无响应,还好能点击开始菜单; 在xp上暂不知晓,暂时没法找到xp环境

0 0