是否有必要使用ScrollWindowEx / ScrollWindow / ScrollDC进行滚动?

时间:2011-12-10 22:16:06

标签: c++ winapi

我正在使用我一直在处理的自定义编辑控件中实现滚动。我想知道的是,使用ScrollWindowEx / ScrollWindow / ScrollDC来实现滚动是必不可少的吗?我看到ScrollWindowEx只是滚动绘画区域。一切都很好,但由于我的编辑控件实现了双缓冲,我还必须更新我的BitBlt。这是一件微不足道的事情,但我想知道它是否必不可少。如果我只使用SetScrollInfo,那也会产生相同的效果。我在这里看到的唯一优势是,当用户向上或向下滚动时,那里已经存在一些文本(因为ScrollWindowEx会移动目标客户区域)并且我不必费心重新绘制。有没有其他优势,或者使用ScrollWindowEx的原因?我是第一次在win32中滚动,这实际上是我第一次独自完成所有处理而不是为我做api,所以我真的不知道如何去做。

P.S。 为了清楚起见,我没有使用MFC。只有Win32 api。 编程语言:非托管C ++

2 个答案:

答案 0 :(得分:5)

您可以以任何方式实现滚动。

ScrollWindow等会滚动客户区的相关部分,并使需要重新绘制的部分无效。

一般来说,这是一种处理滚动的有效而简单的方法,因此很明显它很受欢迎。但是,如果你能证明在你的情况下你可以更有效地获得相同的结果,那就去吧。

答案 1 :(得分:1)

可以在 ScrollCall 中找到一个比较示例。

此外,还有一个有趣的 C 语言示例,使用 ScrollDC 滚动屏幕 here 在该示例中,lprcScrolllprcClip 指的是同一个 RECT,它从滚动中描绘出绘制的输出矩形。

ScrollDC 的绘制输出可以由 WM_PAINT 中的例程处理,只要调用后跟 InvalidateRect。但是,与 ScrollWindow(Ex) 不同,ScrollDC 不关心任何可能覆盖在 DC 中的所有/子/父/兄弟窗口,它更适合单个控件中的图像或文本。要滚动较大的 DC,请务必使用 lprcScrolllprcClip 以避免滚动不可见的 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;
}