关键字: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的消息响应处理过程中分别锁定更新和恢复更新来实现,但在前两种情况下,窗口重绘时涉及两个窗口,就比较麻烦了。 
|