我通过根据滚动条的当前nPos值使WM_PAINT中的所有内容无效并重新绘制来滚动父窗口。我想滚动而不闪烁,所以我处理WM_ERASEBKGND以避免重绘背景。我还为TextOut调用和显示的一些位图执行简单的双缓冲。它的效果很好,除了我的孩子控制。它们会严重闪烁,特别是当nPos == 0且app处理SB_LINEUP或nPos == nMax和app处理SB_LINEDOWN或拖放滚动条时。我用MoveWindow()移动它们。我也尝试过DeferWindowPos()。我用谷歌搜索了一个闪烁的解决方案,但它不起作用或我没有正确使用它。如何消除儿童控件的闪烁?
P.S。当我在主窗口中使用WS_CLIPCHILDREN样式时,控件不会闪烁,但是在滚动窗口的某些部分时会搞砸,特别是我的静态控件,我通过处理WM_DRAWITEM来绘制一些东西。
编辑:(简化代码) 我在WM_PAINT中加倍缓冲,如下所示:
case WM_PAINT: {
hDC = BeginPaint( hwnd, &PS );
hDCMem = CreateCompatibleDC( hDC );
hBMMem = CreateCompatibleBitmap( hDC, wnd_x, wnd_y );
SelectObject( hDCMem, hBMMem );
Font = CreateFont( 130, 50, 0, 0, FW_SEMIBOLD, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, FF_MODERN, "Arial" );
SelectObject( hDCMem, Font );
SetTextColor( hDCMem, RGB( 30, 144, 255 ) );
SetBkColor( hDCMem, RGB( 192, 192, 192 ) );
TextOut( hDCMem, 650, 69 - scr_pos, "ABC", 3 );
MoveWindow( GetDlgItem( hwnd, ID_BUTT_START ), 720, 500 - scr_pos, 160, 150, TRUE );
BitBlt( hDC, 0, 0, wnd_x, wnd_y, hDCMem, 0, 0, SRCCOPY );
DeleteObject( hBMMem );
DeleteDC( hDCMem );
EndPaint( hwnd, &PS );
}
scr_pos是从滚动条获取的当前nPos值。
case WM_ERASEBKGND:
return 1;
break;
case WM_VSCROLL: {
SCROLLINFO sinfo;
ZeroMemory( &sinfo, sizeof( SCROLLINFO ) );
sinfo.cbSize = sizeof( SCROLLINFO );
sinfo.fMask = SIF_POS | SIF_PAGE | SIF_TRACKPOS;
GetScrollInfo( hwnd, SB_VERT, &sinfo );
scr_pos = sinfo.nPos;
switch( LOWORD( wParam ) ) {
case SB_TOP:
scr_pos = 0;
break;
case SB_BOTTOM:
scr_pos = 4000;
break;
case SB_LINEUP: {
scr_pos -= 200;
if ( scr_pos < 0 ) {
scr_pos = 0;
}
}
break;
case SB_LINEDOWN: {
scr_pos += 200;
if ( scr_pos > 4000 ) {
scr_pos = 4000;
}
}
break;
case SB_PAGEUP: {
scr_pos -= si.nPage;
if( scr_pos < 0 ) {
scr_pos = 0;
}
}
break;
case SB_PAGEDOWN: {
scr_pos += si.nPage;
if( scr_pos > 4000 ) {
scr_pos = 4000;
}
}
break;
case SB_THUMBPOSITION:
scr_pos = HIWORD(wParam);
break;
case SB_THUMBTRACK:
scr_pos = HIWORD(wParam);
break;
}
RedrawWindow( hwnd, NULL, NULL, RDW_INVALIDATE | RDW_INTERNALPAINT );
// InvalidateRect ( hwnd, NULL, true );
// UpdateWindow( hwnd );
ZeroMemory( & sinfo, sizeof( SCROLLINFO ) );
sinfo.cbSize = sizeof( SCROLLINFO );
sinfo.fMask = SIF_POS;
snfo.nPos = scr_pos;
SetScrollInfo( hwnd, SB_VERT, & sinfo, TRUE );
}
}
break;
除了儿童控制之外,什么都没有闪烁......
答案 0 :(得分:4)
WS_CLIPCHILDREN
确实是解决方案。继续并激活它,你将不得不追查“我的窗口部分被搞砸了”的原因,但你没有提供足够的细节。
您也可以查看ScrollWindowEx
。该文档专门描述了如何在滚动期间正确地重新定位子窗口。