当前位置:C++技术网 > 精选软件 > Windows零基础入门:2.23 窗口类结构体之窗口类风格CS_NOCLOSE

Windows零基础入门:2.23 窗口类结构体之窗口类风格CS_NOCLOSE

更新时间:2015-09-19 22:49:29浏览次数:1+次

    这节课我们来讲讲下一个窗口类风格CS_NOCLOSE。我是按照窗口类分组介绍的。
    在MSDN中的解释就是禁用了Windows窗口上的关闭系统菜单的关闭菜单项。对于禁用关闭系统菜单的关闭菜单项,你可能就只是以为就是屏蔽了右上角的那个叉叉按钮。其实,我们还需要了解一些其他的知识哦。
    首先我们来看看正常的窗口和禁用关闭菜单项的窗口的区别,见下两张图。

   

            图1:正常窗口

   

            图2:禁用了关闭系统菜单的关闭菜单项的窗口

    从图中可以看到,禁用关闭系统菜单的关闭菜单项,不只是右上角的叉叉按钮不能用了,在单击左上角的标题栏时弹出的应用程序系统菜单也没有了“关闭(C)  Alt+F4”的菜单项,同时也不能使用键盘来快捷关闭了。这三个是同步的,都是产生窗口销毁消息,所以执行的结果都是销毁窗口。所以必须要同步显示和禁用,不然逻辑上就乱套了。

    对于一个窗口,我们有至少三种方式关闭。第一就是右上角,小白都知道的,第二就是单击左上角的标题栏弹出的菜单中单击关闭,或者当前窗口活动时,按下键盘的 Alt+F4就可以快速关闭窗口了。如果最后没有窗口,则会关闭桌面这个窗口,也就意味着关闭操作系统。不然你试试。

    如果你在窗口类风格中加入了CS_NOCLOSE,那么你的窗口就无法直接关闭了,上面的三个方法失效。那么我们既然在窗口上禁用关闭功能,必然要在某种方式中提供一个关闭的途径。酷狗等软件就在系统通知栏中提供了菜单来关闭。

    那么如果程序运行起来后,你需要关闭这个窗口,怎么办呢?当然在正式运行的时候,肯定没有VS了,就不能使用终止调试方式结束了。当然,也不要只能使用任务管理器的方式结束进程,这样可能导致程序内的数据可还来不及保存就终止了。

    为了简化问题,我们就只有一个主窗口,关闭主窗口就让程序结束了。实际上在点击窗口右上角关闭按钮后,窗口接收到了WM_DESTROY消息,在这个消息处理中又推送了一个WM_QUIT消息,就是函数PostQuitMessage执行的,然后程序就结束了。了解到这个机制,我们就可以在系统的关闭按钮禁用后,自己提供关闭程序的实现方法,就是自己推送一个关闭程序的消息WM_QUIT,这样就达到了这个效果。消息处理部分我们会在后面的课程讲到,这里就是简单分析这个相关的窗口类风格涉及到的一点扩展知识。

    那么我们可以在鼠标右击窗口的时候,关闭程序,就如同单击了右上角的叉叉按钮。实现的代码如下:

#include <Windows.h>
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrev, PSTR szCmdLine, int iCmdShow)
{
    HWND hwnd;
    MSG  msg;
    WNDCLASS wndclass;
    wndclass.style =  CS_VREDRAW | CS_HREDRAW | CS_NOCLOSE;
    wndclass.lpfnWndProc = WndProc;
    wndclass.cbClsExtra = 0;
    wndclass.cbWndExtra = 0;
    wndclass.hInstance = hInstance;
    wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
    wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);
    wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
    wndclass.lpszMenuName = NULL;
    wndclass.lpszClassName = L"C1";

    if (!RegisterClass(&wndclass))
    {
        MessageBox(NULL, L"注册窗口类失败,此程序需要运行在Windows NT平台下。", L"C1", MB_ICONERROR);
        return 0;
    }

    hwnd = CreateWindow(L"C1", L"主窗口", WS_OVERLAPPEDWINDOW, 100, 100, 400, 400, NULL, NULL, hInstance, NULL);
    ShowWindow(hwnd, iCmdShow);
    UpdateWindow(hwnd);

    while (GetMessage(&msg, NULL, 0, 0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
    return msg.wParam;
}
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    HDC hdc;
    PAINTSTRUCT ps;
    RECT rect;
    HWND hwnd2;
    switch (message)
    {
    case WM_PAINT:
        hdc = BeginPaint(hwnd, &ps);
        GetClientRect(hwnd, &rect);
        DrawText(hdc, L"主窗口", -1, &rect, DT_CENTER | DT_VCENTER | DT_SINGLELINE);
        SetTextColor(hdc, RGB(0, 0, 255));// - 设置
        EndPaint(hwnd, &ps);
        return 0;
    case WM_RBUTTONDOWN:
        PostQuitMessage(0);
        return 0;
    case WM_DESTROY:
        PostQuitMessage(0);
        return 0;
    }
    return DefWindowProc(hwnd, message, wParam, lParam);
}

     鼠标右击消息为WM_RBUTTONDOWN,你只需要在这个消息中执行一个和窗口销毁的消息一样的代码即可。就是调用函数PostQuitMessage(0); 向消息队列推送一个WM_QUIT消息,这样就达到了目的。如果你对于switch熟悉,其实你可以省掉WM_RBUTTONDOWN的两句代码,直接就执行了WM_DESTROY消息的代码了哦。

    这样鼠标右击窗口,窗口被销毁了。和直接关闭窗口一样的。关闭按钮单击之后,也是产生了WM_DESTROY完成的。那么我们可以主动执行这个代码,一样可以达到效果,是不是很有意思呢。你还可以试试各种方式来关闭窗口哦。试试看吧。

    这里需要注明一点,关闭窗口并不表示关闭程序。关闭窗口只是将窗口最小化了而已。要销毁窗口,就是要让窗口的消息循环结束,这样就彻底关闭了,而不是最小化。这个是很多人的误区。而窗口销毁消息就是来推送WM_QUIT消息来终止窗口消息循环的,这样就可以结束程序。当然,是否要在销毁窗口消息中终止程序,还是你说了算。你可以在WM_DESTROY消息中不终止程序,而是隐藏窗口,调用ShowWindow(hwnd,SW_HIDE)函数即可,而在通知栏里右击菜单,提供关闭程序的途径。这个是很多软件,比如酷狗等采用的方案。实际在实现上很简单,就是这么回事。至于通知栏关闭,就不是现在讨论的问题咯。

    总结一下,我们使用CS_NOCLOSE窗口类风格,不只是简单的去掉了右上角的关闭窗口按钮,还删掉了系统菜单的关闭菜单项和禁用快捷键。同时我们也了解到单击关闭按钮内部执行了哪些动作,从而我们可以自己来实现,以达到一个个性化的方案。最后就谈到了关闭窗口并不等于结束程序,很多软件就让你关闭窗口程序依然在运行,道理就是如此。