E=MC2 发表于 2014-6-5 12:20:04

在第6讲视频中的WM_NCLBUTTONDOWN用法问题

1,窗口的小X字图标发送的消息和WM_CLOSE消息不是一个消息,所以我要问的是小X子关闭窗口的消息名是什么?WM_???
2,从甲鱼老师视频中可以看出系统是优先发送WM_NCLBUTTONDOWN消息而不发送关闭窗口WM_???消息,那么问题来了,如何在我点下鼠标左键后优先发送1的WM_??消息而不发送WM_NCLBUTTONDOWN消息,以达到点除小X字外的非客户区都能出现视频里面的提示窗口,而点小X子就会关闭窗口?

仰望天上的光 发表于 2014-6-5 12:20:05

E=MC2 发表于 2014-7-16 11:35
学习了。但是问题2貌似还是无解啊?

#include <windows.h>

#include <stdio.h>


LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;


int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,

                  PSTR szCmdLine, int iCmdShow)
{
      static TCHAR szAppName[] = TEXT ("HelloWin") ;
      HWND         hwnd ;
      MSG          msg ;
      WNDCLASS   wndclass ;   
      wndclass.style         = CS_HREDRAW | CS_VREDRAW ;
      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 = szAppName ;
      if (!RegisterClass (&wndclass))
      {
                MessageBox (NULL, TEXT ("This program requires Windows NT!"),
                        szAppName, MB_ICONERROR) ;
                return 0 ;
      }
      hwnd = CreateWindow (szAppName,                  // window class name
                TEXT ("The Hello Program"), // window caption
                WS_OVERLAPPEDWINDOW,      // window style
                CW_USEDEFAULT,            // initial x position
                CW_USEDEFAULT,            // initial y position
                CW_USEDEFAULT,            // initial x size
                CW_USEDEFAULT,            // initial y size
                NULL,                     // parent window handle
                NULL,                     // window menu handle
                hInstance,                  // program instance handle
                NULL) ;                     // creation parameters
      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)
{
      switch (message)
      {
      case WM_CREATE:
                return 0 ;
                case        WM_NCLBUTTONDOWN:
                        {
                                if( wParam != HTCLOSE ) {
                                        MessageBox(hwnd, "按到我了", "Msg", 0 );
                                        return 1;
                                }
                                break;
                        }
      case WM_SYSCOMMAND:
                if( wParam == SC_CLOSE )
                {
                        MessageBox(hwnd, "WM_SYSCOMMAND,窗口还在,还可以撤销操作", "Msg", 0 );
                        SendMessage( hwnd, WM_CLOSE, 0, 0 );
                }
                return 0;
      case WM_CLOSE:
                {
                        MessageBox(hwnd, "WM_CLOSE,窗口还在,还可以撤销操作", "Msg", 0 );
                        DestroyWindow( hwnd );
                        return 0;
                }
      case WM_DESTROY:
                {
                        MessageBox(hwnd, "WM_DESTROY,窗口不在,不可以撤销操作", "Msg", 0 );
                        PostQuitMessage (0) ;
                        return 0 ;
                }
      }
      return DefWindowProc (hwnd, message, wParam, lParam) ;
}

ravenhu13 发表于 2014-6-5 14:26:41

用setwindowhook,不过貌似已经超出你目前的能力范围了。

メ㊣逆ご帅☆ 发表于 2014-6-5 19:43:24

本帖最后由 メ㊣逆ご帅☆ 于 2014-6-5 19:55 编辑

窗口小X肯定会发送WM_CLOSE消息的
因为我的程序就是在处理WM_CLOSE消息的窗口过程中做收尾工作的
case WM_CLOSE:
                {
                     //收尾工作
                }
然而每次都是成功收尾,代表WM_CLOSE消息有被捕获。而且我每次都是按小X关闭程序的
所以正常的小X肯定会发送WM_CLOSE

如果你说的是视频中响应的是什么消息,当然是WM_NCLBUTTONDOWN


关于你的第二个问题
:如何在我点下鼠标左键后优先发送1的WM_??消息而不发送WM_NCLBUTTONDOWN消息,以达到点除小X字外的非客户区都能出现视频里面的提示窗口,而点小X子就会关闭窗口?
如果要实现。
你只要在消息循环中添加WM_NCLBUTTONDOWN的处理过程。

取鼠标坐标
然后判断是否属于小X坐标范围,如果属于继续发送WM_CLOSE消息,如果不属于就显示提示窗口

伪代码:case WM_NCLBUTTONDOWN:
                {                         //取鼠标坐标                        if(鼠标坐标属于小X按钮范围)                        {                                 //发送WM_CLOSE消息                        }                        else                        {                              //显示提示窗口                        }                }这是方法中的一个而已。WINDOWS程序设计非常灵活。之后你接触多了应该会深有感触、

青玄 发表于 2014-6-5 19:51:00

恩恩!楼上的这位兄弟说的很全面!支持!

E=MC2 发表于 2014-6-5 21:05:13

メ㊣逆ご帅☆ 发表于 2014-6-5 19:43 static/image/common/back.gif
窗口小X肯定会发送WM_CLOSE消息的
因为我的程序就是在处理WM_CLOSE消息的窗口过程中做收尾工作的
case WM ...

首先我赞同你对于第二个问题的回答!其次关于第一个问题,我还是有疑问!你给CALLBACKWndProc循环里添加如下代码:case WM_NCLBUTTONDOWN:
                MessageBox(hwnd, TEXT("点到我了!"), TEXT("舒服"), MB_OK);
                return 0;
        case WM_CLOSE:
                MessageBox(hwnd, TEXT("不要关闭额"), TEXT("哎~"), MB_OK);
                return 0;你会发现在点小X子时,会提示“点到我了”,响应的是WM_NCLBUTTONDOWN消息。而点桌面最下面的窗口图标关闭窗口时,会提示“不要关闭额”,响应的是WM_CLOSE消息。

メ㊣逆ご帅☆ 发表于 2014-6-5 21:10:15

本帖最后由 メ㊣逆ご帅☆ 于 2014-6-5 21:11 编辑

E=MC2 发表于 2014-6-5 21:05 static/image/common/back.gif
首先我赞同你对于第二个问题的回答!其次关于第一个问题,我还是有疑问!你给CALLBACKWndProc循环里添加 ...
WM_NCLBUTTONDOWN是左键单击该程序的非客户区响应的消息
例如标题栏等,如果单击小X,优先触发WM_NCLBUTTONDOWN
这时会不发送WM_CLOSE

而点桌面最下面的窗口图标关闭窗口时,不属于本程序的非客户区
所以不会发送WM_NCLBUTTONDOWN,这时没有消息与WM_CLOSE冲突
于是正常发送WM_CLOSE
所以响应WM_CLOSE

E=MC2 发表于 2014-6-5 21:50:55

本帖最后由 E=MC2 于 2014-6-5 22:03 编辑

メ㊣逆ご帅☆ 发表于 2014-6-5 21:10 static/image/common/back.gif
WM_NCLBUTTONDOWN是左键单击该程序的非客户区响应的消息
例如标题栏等,如果单击小X,优先触发WM_NCLBUT ...

我刚才又在网上查了下,关于问题二。您说可以用鼠标获取坐标的方式来处理。但是我在网上只找到了获取客户区鼠标的相关信息,获取非客户区鼠标的函数名叫啥啊? 查不到额。有这些函数GetCursor,GetClipCursor,ClipCursor,GetPhysicalCursorPos,GetCursorPos 这五个函数。我只知道客户的英文是client.

メ㊣逆ご帅☆ 发表于 2014-6-5 23:33:42

E=MC2 发表于 2014-6-5 21:50 static/image/common/back.gif
我刚才又在网上查了下,关于问题二。您说可以用鼠标获取坐标的方式来处理。但是我在网上只找到了获取客 ...




GetCursorPos
获取鼠标坐标(以整个显示器左上角为(0,0)原点,就是屏幕坐标)。

GetMenuItemRect获取标题栏坐标
判断当前鼠标坐标是否属于标题栏右边一个按钮大小的范围内
即鼠标X坐标<标题栏X坐标且鼠标X坐标>(标题栏X坐标-按钮宽度大小)
鼠标Y坐标<标题栏Y坐标且鼠标Y坐标>(标题栏Y坐标-按钮高度大小)

这里有个问题就是我无法获取到SC_CLOSE这个系统按钮的宽高度。(如果是普通控件按钮还比较轻松。)
目前只能借助拖一个按钮控件。目测跟X按钮大小差不多,然后看该按钮大小为多少。应该跟小X按钮大小差不多


是个有缺陷的想法了。你自己想其他办法去吧






小甲鱼 发表于 2014-6-8 14:41:53

E=MC2 发表于 2014-6-5 21:50 static/image/common/back.gif
我刚才又在网上查了下,关于问题二。您说可以用鼠标获取坐标的方式来处理。但是我在网上只找到了获取客 ...

不用专门去获取,WM_NCLBUTTONDOWN 消息的 lParam 成员就是非客户区触发此消息时候的鼠标指针。

资料传送门:http://msdn.microsoft.com/en-us/library/windows/desktop/ms645620(v=vs.85).aspx

E=MC2 发表于 2014-6-8 19:08:50

小甲鱼 发表于 2014-6-8 14:41 static/image/common/back.gif
不用专门去获取,WM_NCLBUTTONDOWN 消息的 lParam 成员就是非客户区触发此消息时候的鼠标指针。

资料传 ...

在您给的网址上没有找到相关信息。下面这个代码就是利用了lParam 成员来获取鼠标坐标的,但是不能获取非客户区的坐标。/* -------------------------------------------------------------------
                  MyWindows.c -- 基本窗口模型
                                《Windows 程序设计(SDK)》视频教程                  
--------------------------------------------------------------------*/

#include <windows.h>
#include<windowsx.h>

LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);            
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow)
{
        static TCHAR szAppName[] = TEXT("MyWindows");
        HWND hwnd;
        MSG msg;
        WNDCLASS wndclass;

        wndclass.style = CS_HREDRAW | CS_VREDRAW;
        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 = szAppName;

        if (!RegisterClass(&wndclass))
        {
                MessageBox(NULL, TEXT("这个程序需要在 Windows NT 才能执行!"), szAppName, MB_ICONERROR);
                return 0;
        }

        hwnd = CreateWindow(szAppName,
                TEXT("鱼C工作室"),
                WS_OVERLAPPEDWINDOW,
                CW_USEDEFAULT,
                CW_USEDEFAULT,
                CW_USEDEFAULT,
                CW_USEDEFAULT,
                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;
        TCHAR szBuffer;
        POINT point;
        RECT rect;
        static RECT rr;
        static int cxClient;
        static int cyClient;

        switch (message){

        case WM_SIZE:
                cxClient = GET_X_LPARAM(lParam);
                cyClient = GET_Y_LPARAM(lParam);
                GetClientRect(hwnd, &rr);
                //wsprintf(szBuffer,TEXT("cyClient %d"),cyClient);
                //MessageBox(NULL,szBuffer,TEXT("tip"),MB_ICONERROR);

                return 0;

        case WM_PAINT:
                hdc = BeginPaint(hwnd, &ps);

                GetCursorPos(&point);
                TextOut(hdc, 200, 200, szBuffer, wsprintf(szBuffer, TEXT("the point is %d,%d"), point.x, point.y));
                EndPaint(hwnd, &ps);
                return 0;

        case WM_DESTROY:
                PostQuitMessage(0);
                return 0;

        case WM_MOUSEMOVE:
                //SendMessage(hwnd,WM_PAINT,0,0);
                rect.left = rr.left;
                rect.top = rr.top;
                rect.right = rr.right;
                rect.bottom = rr.bottom;

                InvalidateRect(hwnd, &rect, TRUE);
                return 0;

        }


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

小甲鱼 发表于 2014-6-8 19:39:26

E=MC2 发表于 2014-6-8 19:08 static/image/common/back.gif
在您给的网址上没有找到相关信息。下面这个代码就是利用了lParam 成员来获取鼠标坐标的,但是不能获取非客 ...

哥们给翻译下:

Posted when the user presses the left mouse button while the cursor is within the nonclient area of a window. This message is posted to the window that contains the cursor. If a window has captured the mouse, this message is not posted.

......

lParam

A POINTS structure that contains the x- and y-coordinates of the cursor. The coordinates are relative to the upper-left corner of the screen.

小甲鱼 发表于 2014-6-8 19:50:32

E=MC2 发表于 2014-6-8 19:08 static/image/common/back.gif
在您给的网址上没有找到相关信息。下面这个代码就是利用了lParam 成员来获取鼠标坐标的,但是不能获取非客 ...

另外贴段代码供参考:/* -------------------------------------------------------------------
MyWindows.c -- 基本窗口模型
《Windows 程序设计(SDK)》视频教程
--------------------------------------------------------------------*/

#include <windows.h>
#include <Windowsx.h>

LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);

int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow)
{
        static TCHAR szAppName[] = TEXT("MyWindows");
        HWND hwnd;
        MSG msg;
        WNDCLASS wndclass;

        wndclass.style = CS_HREDRAW | CS_VREDRAW;
        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 = szAppName;

        if (!RegisterClass(&wndclass))
        {
                MessageBox(NULL, TEXT("这个程序需要在 Windows NT 才能执行!"), szAppName, MB_ICONERROR);
                return 0;
        }

        hwnd = CreateWindow(szAppName,
                TEXT("鱼C工作室"),
                WS_OVERLAPPEDWINDOW,
                CW_USEDEFAULT,
                CW_USEDEFAULT,
                CW_USEDEFAULT,
                CW_USEDEFAULT,
                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;
        TCHAR szBuffer;

        switch (message)
        {
        case WM_PAINT:
                hdc = BeginPaint(hwnd, &ps);
                GetClientRect(hwnd, &rect);
                DrawText(hdc, TEXT("Hello world!"), -1, &rect,
                        DT_SINGLELINE | DT_CENTER | DT_VCENTER);
                EndPaint(hwnd, &ps);
                return 0;

        case WM_LBUTTONUP:
                MessageBox(hwnd, TEXT("哎呀,我丫的被按了一下~"), TEXT("好舒服~"), MB_OK);
                return 0;

        case WM_NCLBUTTONDOWN:
                wsprintf(szBuffer, TEXT("the point is %d,%d"), GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam));
                MessageBox(hwnd, szBuffer, TEXT("哎呀,我丫的被按了一下~"), MB_OK);
                return 0;

        case WM_DESTROY:
                PostQuitMessage(0);
                return 0;
        }

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

E=MC2 发表于 2014-6-9 09:53:02

小甲鱼 发表于 2014-6-8 19:39 static/image/common/back.gif
哥们给翻译下:

当光标在窗口的非客户区时,使用者点击鼠标左键,此消息被发送到一个包含光标的窗口。如果一个窗口捕获了鼠标,这个消息不被发送。(captured词该翻译成什么? 我不太清楚,貌似“捕获”一次不能让我理解。)
......
一个POINT结构包含光标x与y的坐标。该坐标是以屏幕的左上角为原点来计算的。

E=MC2 发表于 2014-6-9 09:55:25

本帖最后由 E=MC2 于 2014-6-9 10:20 编辑

1,主要是没有看到“the nonclient area ”这个词。
2,我还发现了,每次打开窗口的时候,同一位置捕获的窗口坐标是不一样的。不清楚这是什么原因?(可能每次窗口在屏幕中的位置都发生变化引起的吧? 如果是这样,那么就要设计一个左键或右键点击鼠标不放来拖动窗口的消息来挪动窗口回到原来的位置,这样才能在代码判断的区域来关闭窗口) 总之这种捕获鼠标坐标的方法不太可行额。还是要想想其他办法。。

myq549277513 发表于 2014-6-11 07:10:49

最近对Windows编程比较感兴趣!

myq549277513 发表于 2014-6-11 07:11:29

这好像是非客户区的消息~

爱裸奔的箱子 发表于 2014-6-13 01:02:04

我最近看MFC,我对的理解是,一个发送的鼠标消息 告诉系统我在那个坐标点击了一下,然后系统知道那个地方是X所以系统又间接给程序发送了WM_CLOSE的命令。如果没记错的话应该是DispatchMessage函数起的作用。都是大白话和自己的理解,我不是大神更不是学霸,只是一个初级小乌龟……讲错的话请大家理解一下……

小科来啦 发表于 2014-7-10 00:40:20

1、从小X关闭窗口的产生的消息名是WM_DESTROY
2、将return 0;改为break;点第二次X时可将窗口关闭,具体原因还不知道,没调试成,搞了一个小时了快

E=MC2 发表于 2014-7-10 08:58:09

小科来啦 发表于 2014-7-10 00:40
1、从小X关闭窗口的产生的消息名是WM_DESTROY
2、将return 0;改为break;点第二次X时可将窗口关闭,具体原 ...

确实按2次小X即可关闭,那么是什么原因呢?
页: [1] 2
查看完整版本: 在第6讲视频中的WM_NCLBUTTONDOWN用法问题