为什么在将应用程序窗口侧拖得快一点时听到哔声需要这么长时间?

时间:2013-08-20 21:39:40

标签: windows winapi

这是一个使用Windows API的非常简单的代码,其中子窗口绘制在应用程序窗口的中央部分。通过在主窗口客户区域上单击鼠标左键,子窗口将采用空高度。通过单击鼠标右键,子窗口的高度在每次单击鼠标时增加10个像素。由于在绘制父窗口时进行了MessageBeep(-1)调用,每次在窗口的应用程序中发生左键或右键单击时,都可以听到哔声。

奇怪的是,当您拖动父窗口的一侧,改变其宽度或高度时,非常慢,您可以在每次移动时听到哔哔声。但是如果你将窗口的侧面拖得快一点,那么在释放鼠标按钮后,你只会听到一声嘟嘟声,很多秒。那是为什么?

这是代码:

#include <windows.h>

LRESULT CALLBACK        WindowProc(HWND, UINT, UINT, LONG);
LRESULT CALLBACK        ChildProc(HWND, UINT, UINT, LONG);

/****************************************************************************************************************************

    WinMain()

****************************************************************************************************************************/

int APIENTRY WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpszCmdLine, int nCmdShow)
{
    MSG         msg;
    WNDCLASSEX  wndclassx;

    wndclassx.cbSize        = sizeof(WNDCLASSEX);
    wndclassx.style         = 0;
    wndclassx.lpfnWndProc   = WindowProc;
    wndclassx.cbClsExtra    = 0;
    wndclassx.cbWndExtra    = 0;
    wndclassx.hInstance     = hInstance;
    wndclassx.hIcon         = 0;
    wndclassx.hCursor       = LoadCursor(NULL, IDC_ARROW);
    wndclassx.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
    wndclassx.lpszMenuName  = NULL;
    wndclassx.lpszClassName = L"ParentWindow";
    wndclassx.hIconSm       = NULL;

    if( !RegisterClassEx(&wndclassx) ) return 0;

    wndclassx.cbSize        = sizeof(WNDCLASSEX);
    wndclassx.style         = 0;
    wndclassx.lpfnWndProc   = ChildProc;
    wndclassx.cbClsExtra    = 0;
    wndclassx.cbWndExtra    = 0;
    wndclassx.hInstance     = hInstance;
    wndclassx.hIcon         = 0;
    wndclassx.hCursor       = 0;
    wndclassx.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
    wndclassx.lpszMenuName  = NULL;
    wndclassx.lpszClassName = L"ChildWindow";
    wndclassx.hIconSm       = NULL;

    if( !RegisterClassEx(&wndclassx) ) return 0;

    HWND hWnd;

    if( !(hWnd = CreateWindow(L"ParentWindow", L"Parent Window", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT,
                              CW_USEDEFAULT, CW_USEDEFAULT, NULL, NULL, hInstance, NULL)) ) return 0;

    ShowWindow(hWnd, SW_SHOW);
    UpdateWindow(hWnd);

    while( GetMessage(&msg, NULL, 0, 0) )
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }

    return (int)msg.wParam;
}

/****************************************************************************************************************************

    WindowProc()

****************************************************************************************************************************/

LRESULT CALLBACK WindowProc (HWND hwnd, UINT message, UINT wParam, LONG lParam)
{
    PAINTSTRUCT ps;

    switch ( message )
    {
        RECT rect;
        HWND    hChild;

        case WM_CREATE:

        GetClientRect(hwnd, &rect);

        if( !CreateWindow(L"ChildWindow", NULL, WS_CHILD | WS_VISIBLE | WS_BORDER, rect.right / 3, rect.bottom / 3, rect.right / 3,
                          rect.bottom / 3, hwnd, (HMENU)0, ((LPCREATESTRUCT)lParam)->hInstance, NULL) ) return -1;
        break;

        case WM_SIZE:

        MoveWindow(GetDlgItem(hwnd, 0), LOWORD(lParam) / 3, HIWORD(lParam) / 3, LOWORD(lParam) / 3, HIWORD(lParam) / 3,
                   true);

        break;

        case WM_LBUTTONDOWN:

        hChild = GetDlgItem(hwnd, 0);

        GetWindowRect(hChild, &rect);


        ScreenToClient(hwnd, (LPPOINT)&rect);


        ScreenToClient(hwnd, (LPPOINT)&rect.right);

        MoveWindow(hChild, rect.left, rect.top, rect.right - rect.left, 0, true);
        break;

        case WM_RBUTTONDOWN:

        hChild = GetDlgItem(hwnd, 0);

        GetWindowRect(hChild, &rect);

        ScreenToClient(hwnd, (LPPOINT)&rect);

        ScreenToClient(hwnd, (LPPOINT)&rect.right);

        MoveWindow(hChild, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top + 10, true);
        break;

        case WM_PAINT:
        BeginPaint(hwnd, &ps);
        MessageBeep(-1);
        EndPaint(hwnd, &ps);
        break;

        case WM_DESTROY:

        PostQuitMessage(0);
        break;

        default:

        return DefWindowProc(hwnd, message, wParam, lParam);
    }
    return 0;
}

/****************************************************************************************************************************

    ChildProc()

****************************************************************************************************************************/

LRESULT CALLBACK ChildProc (HWND hwnd, UINT message, UINT wParam, LONG lParam)
{
    switch( message )
    {
        default:

        return DefWindowProc(hwnd, message, wParam, lParam);
    }
}

1 个答案:

答案 0 :(得分:1)

引用MSDN上的KB article

  

正常使用GetMessage()(为所有参数传递零   除了LPMSG参数)或PeekMessage()之外,还有任何消息   在用户输入消息之前处理应用程序队列。并输入   消息在WM_TIMERWM_PAINT“消息之前处理。”

换句话说,因为你在MessageBeep期间调用了WM_PAINT,所以直到窗口停止处理用户输入之后才会发生这种情况,例如,当用户停止移动窗口时。