关键字:MFC, MDI, Flicker
用Visual Studio的App Wizard创建MDI项目后,我们会发现在子窗口(CMDIChildWnd)处于最大化状态时常会发生闪烁现象(尤其是内嵌浏览器时),一般说来有如下几种情况:
1、当前子窗口处于最大化状态时创建新的窗口,会看到一个矩形闪烁的过程。
2、切换窗口时现象同上(调用MDINext和MDIPrev不会出现闪烁,而调用MDIActivate则会引起闪烁)。
3、如果MainFrame上有其它停靠的面板(CControlBar),则调用ShowControlBar显示/隐藏面板时,子窗口边界会有闪烁现象。
解决的方法:
1、重载PreCreateWindow,此方法能够解决上述前两种情况的闪烁问题:
BOOL CChildFrame::PreCreateWindow(CREATESTRUCT& cs)
{
// TODO: Modify the Window class or styles here by modifying
// the CREATESTRUCT cs
cs.style = WS_CHILD | WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU
| FWS_ADDTOTITLE | WS_THICKFRAME | WS_MINIMIZEBOX | WS_MAXIMIZEBOX
| WS_MAXIMIZE; //把窗口样式设置为最大化,但先不显示,问题1
cs.cx = 1024; //把窗口大小设置为整个屏幕大小,问题2
cs.cy = 768;
if( !CMDIChildWndEx::PreCreateWindow(cs) )
return FALSE;
cs.style |= WS_VISIBLE; //创建完成只之后再显示窗口,问题1
return TRUE;
}
2、第3个问题的方法是处理MainFrame的MDI client区域,MainFrame(CMDIFrameWnd)的的MDI client窗口句柄保存在成员变量m_hWndMDIClient中,只需要在调用ShowControlBar之前将该窗口的更新锁定,调用完ShowControlBar后再恢复更新即可消除闪烁,方法可以解决所有子窗口改变引起的闪烁,例子如下:
// 切换显示左边的面板
void CMainFrame::ToggleLeftPane(CControlBar & pBar)
{
::SendMessage(m_hWndMDIClient, WM_SETREDRAW, FALSE, 0L);
ShowControlBar( &pBar, !pBar.IsVisible(), FALSE );
::SendMessage(m_hWndMDIClient, WM_SETREDRAW, TRUE, 0L);
::InvalidateRect(m_hWndMDIClient, NULL, TRUE);//强制重绘
}
注意此处不应调用::LockWindowUpdate()来锁定和恢复更新,LockWindowUpdate在恢复更新时会造成除宿主窗口外整个屏幕的窗口轻微闪烁。
调用MDIActivate切换子窗口引起的闪烁同样可以解决:
......
::SendMessage(m_hWndMDIClient, WM_SETREDRAW, FALSE, 0L);
MDIActivate( pTmpChild );
::SendMessage(m_hWndMDIClient, WM_SETREDRAW, TRUE, 0L);
::InvalidateRect(m_hWndMDIClient, NULL, TRUE);//强制重绘
......
至于原理,我认为是封装的缘故,为了保证窗口的内容的更新,重绘的操作时时在发生,如果重绘不够快且每个窗口都在重绘的话,闪烁就产生了,所以适当的时候禁止窗口重绘,适当的时候再恢复,就可以解决问题。
不过总是在操作窗口之前锁定更新,操作之后又恢复更新麻烦了一点,似乎应该能够找到两个合适的位置来放置这两句话,起到一劳永逸的效果。对于第三种情况,可以简单地通过子类化m_hWndMDIClient,在WM_WINDOWPOSCHANGING和WM_WINDOWPOSCHANGED的消息响应处理过程中分别锁定更新和恢复更新来实现,但在前两种情况下,窗口重绘时涉及两个窗口,就比较麻烦了。