我正在使用我一直在处理的自定义编辑控件中实现滚动。我想知道的是,使用ScrollWindowEx / ScrollWindow / ScrollDC来实现滚动是必不可少的吗?我看到ScrollWindowEx只是滚动绘画区域。一切都很好,但由于我的编辑控件实现了双缓冲,我还必须更新我的BitBlt。这是一件微不足道的事情,但我想知道它是否必不可少。如果我只使用SetScrollInfo,那也会产生相同的效果。我在这里看到的唯一优势是,当用户向上或向下滚动时,那里已经存在一些文本(因为ScrollWindowEx会移动目标客户区域)并且我不必费心重新绘制。有没有其他优势,或者使用ScrollWindowEx的原因?我是第一次在win32中滚动,这实际上是我第一次独自完成所有处理而不是为我做api,所以我真的不知道如何去做。
P.S。 为了清楚起见,我没有使用MFC。只有Win32 api。 编程语言:非托管C ++
答案 0 :(得分:5)
您可以以任何方式实现滚动。
ScrollWindow
等会滚动客户区的相关部分,并使需要重新绘制的部分无效。
一般来说,这是一种处理滚动的有效而简单的方法,因此很明显它很受欢迎。但是,如果你能证明在你的情况下你可以更有效地获得相同的结果,那就去吧。
答案 1 :(得分:1)
可以在 ScrollCall 中找到一个比较示例。
此外,还有一个有趣的 C 语言示例,使用 ScrollDC 滚动屏幕 here
在该示例中,lprcScroll
和 lprcClip
指的是同一个 RECT,它从滚动中描绘出绘制的输出矩形。
ScrollDC 的绘制输出可以由 WM_PAINT 中的例程处理,只要调用后跟 InvalidateRect
。但是,与 ScrollWindow(Ex) 不同,ScrollDC 不关心任何可能覆盖在 DC 中的所有/子/父/兄弟窗口,它更适合单个控件中的图像或文本。要滚动较大的 DC,请务必使用 lprcScroll
和 lprcClip
以避免滚动不可见的 DC 区域。
如前所述,ScrollWindoW(Ex) 是首选,特别是对于具有混合控件内容的窗口。
为了进一步说明,包括从 here 复制的所有三个函数的令人惊奇的古老实现:
* Scroll windows and DCs
*
* Copyright David W. Metcalfe, 1993
*
*/
static char Copyright[] = "Copyright David W. Metcalfe, 1993";
#include <stdlib.h>
#include "windows.h"
#include "gdi.h"
#include "stddebug.h"
/* #define DEBUG_SCROLL /* */
/* #undef DEBUG_SCROLL /* */
#include "debug.h"
static int RgnType;
/*************************************************************************
* ScrollWindow (USER.61)
*/
void ScrollWindow(HWND hwnd, short dx, short dy, LPRECT rect, LPRECT clipRect)
{
HDC hdc;
HRGN hrgnUpdate;
RECT rc, cliprc;
dprintf_scroll(stddeb,"ScrollWindow: dx=%d, dy=%d, rect=%d,%d,%d,%d\n",
dx, dy, rect->left, rect->top, rect->right, rect->bottom);
hdc = GetDC(hwnd);
if (rect == NULL)
GetClientRect(hwnd, &rc);
else
CopyRect(&rc, rect);
if (clipRect == NULL)
GetClientRect(hwnd, &cliprc);
else
CopyRect(&cliprc, clipRect);
hrgnUpdate = CreateRectRgn(0, 0, 0, 0);
ScrollDC(hdc, dx, dy, &rc, &cliprc, hrgnUpdate, NULL);
InvalidateRgn(hwnd, hrgnUpdate, TRUE);
ReleaseDC(hwnd, hdc);
}
/*************************************************************************
* ScrollDC (USER.221)
*/
BOOL ScrollDC(HDC hdc, short dx, short dy, LPRECT rc, LPRECT cliprc,
HRGN hrgnUpdate, LPRECT rcUpdate)
{
HRGN hrgnClip, hrgn1, hrgn2;
POINT src, dest;
short width, height;
DC *dc = (DC *)GDI_GetObjPtr(hdc, DC_MAGIC);
dprintf_scroll(stddeb, "ScrollDC: dx=%d, dy=%d, rc=%d,%d,%d,%d\n", dx, dy,
rc->left, rc->top, rc->right, rc->bottom);
if (rc == NULL)
return FALSE;
if (cliprc)
{
hrgnClip = CreateRectRgnIndirect(cliprc);
SelectClipRgn(hdc, hrgnClip);
}
if (dx > 0)
{
src.x = XDPTOLP(dc, rc->left);
dest.x = XDPTOLP(dc, rc->left + abs(dx));
}
else
{
src.x = XDPTOLP(dc, rc->left + abs(dx));
dest.x = XDPTOLP(dc, rc->left);
}
if (dy > 0)
{
src.y = YDPTOLP(dc, rc->top);
dest.y = YDPTOLP(dc, rc->top + abs(dy));
}
else
{
src.y = YDPTOLP(dc, rc->top + abs(dy));
dest.y = YDPTOLP(dc, rc->top);
}
width = rc->right - rc->left - abs(dx);
height = rc->bottom - rc->top - abs(dy);
if (!BitBlt(hdc, dest.x, dest.y, width, height, hdc, src.x, src.y,
SRCCOPY))
return FALSE;
if (hrgnUpdate)
{
if (dx > 0)
hrgn1 = CreateRectRgn(rc->left, rc->top, rc->left+dx, rc->bottom);
else if (dx < 0)
hrgn1 = CreateRectRgn(rc->right+dx, rc->top, rc->right,
rc->bottom);
else
hrgn1 = CreateRectRgn(0, 0, 0, 0);
if (dy > 0)
hrgn2 = CreateRectRgn(rc->left, rc->top, rc->right, rc->top+dy);
else if (dy < 0)
hrgn2 = CreateRectRgn(rc->left, rc->bottom+dy, rc->right,
rc->bottom);
else
hrgn2 = CreateRectRgn(0, 0, 0, 0);
RgnType = CombineRgn(hrgnUpdate, hrgn1, hrgn2, RGN_OR);
}
if (rcUpdate) GetRgnBox( hrgnUpdate, rcUpdate );
return TRUE;
}
/*************************************************************************
* ScrollWindowEx (USER.319)
*/
int ScrollWindowEx(HWND hwnd, short dx, short dy, LPRECT rect, LPRECT clipRect,
HRGN hrgnUpdate, LPRECT rcUpdate, WORD flags)
{
HDC hdc;
RECT rc, cliprc;
dprintf_scroll(stddeb,"ScrollWindowEx: dx=%d, dy=%d, rect=%d,%d,%d,%d\n",
dx, dy, rect->left, rect->top, rect->right, rect->bottom);
hdc = GetDC(hwnd);
if (rect == NULL)
GetClientRect(hwnd, &rc);
else
CopyRect(&rc, rect);
if (clipRect == NULL)
GetClientRect(hwnd, &cliprc);
else
CopyRect(&cliprc, clipRect);
ScrollDC(hdc, dx, dy, &rc, &cliprc, hrgnUpdate, rcUpdate);
if (flags | SW_INVALIDATE)
{
RedrawWindow(hwnd, NULL, hrgnUpdate,
RDW_INVALIDATE | ((flags & SW_ERASE) ? RDW_ERASENOW : 0));
}
ReleaseDC(hwnd, hdc);
return RgnType;
}