MFC控件编程之鼠标跟键盘消息

时间:2022-06-21
本文章向大家介绍MFC控件编程之鼠标跟键盘消息,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

            MFC控件编程之鼠标跟键盘消息

在MFC中鼠标消息.键盘消息我们很常用.所以说一下.

鼠标消息分为客户区消息.跟非客户区消息.

一丶客户区消息

我们可以处理消息.来进行我们相应的函数即可.

MFC添加消息的话.可以自己在消息映射表中添加 .可以自己使用工具直接添加.

消息映射表添加

学习过MFC原理的应该知道.这里可以添加消息. 自己可以手动添加.我们也可以使用向导来添加.

void CCDCDlg::OnLButtonDown(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值

    CString str;
    str.Format(TEXT("x坐标 = %d,y 坐标 = %d"), point.x, point.y);
    ::SetWindowText(m_hWnd, str);
}

输出我们的鼠标点击的时候的坐标位置.调用 ::SetWindowText设置窗口标题.

三丶非客户区消息

非客户区消息就指的标题栏啊等等.这些位置.

而他们的消息多了一个NC开头.

例如我们在消息映射表中添加一个NC开头的消息.

我们也可以使用向导.看自己喜好了.

代码:

void CCDCDlg::OnNcLButtonDown(UINT nFlags, CPoint point)
{
    CString str;
    str.Format(TEXT("非客户区 x坐标 = %d,y 坐标 = %d"), point.x, point.y);
    ::SetWindowText(m_hWnd, str);
}

点击标题栏

如果按照上面写我们点击关闭按钮会关闭不了.原因就是NcLBUTTONDown里面包括了点击关闭按钮的消息.你没有进行处理.此时我们交给父类处理即可.

void CCDCDlg::OnNcLButtonDown(UINT nFlags, CPoint point)
{
    CString str;
    str.Format(TEXT("非客户区 x坐标 = %d,y 坐标 = %d"), point.x, point.y);
    ::SetWindowText(m_hWnd, str);
    CDialogEx::OnNcLButtonDown(nFlags,point);
}

其中nflags是一个区域划分吗.我们可以自己判断是否是哪个区域会享用.

例如:

  if (nflags == HTSYSMENU) 判断是否是菜单.... 进行你的操作

点击客户区

在我们的鼠标消息到来之前.它会先判断是客户区域还是非客户区域.然后进行转换.才给我们将消息发送过来.

也就是说我们可以吧客户区转换为非客户区. 比如鼠标点击客户区.其实是点击的非客户区.

代码如下.

LRESULT CCDCDlg::OnNcHitTest(CPoint point)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值

    UINT hit = CDialogEx::OnNcHitTest(point); //获取父类返回的判断是否是客户区还是非客户区
    if (hit == HTCLIENT) //如果是客户区.那么我们转化为非客户区
    {
        hit = HTCAPTION;
    }
    return hit;
}

实现的消息

API: 坐标转为窗口坐标.

::ScreenToClient(m_hWnd, &point);

四丶捕获鼠标

捕获鼠标也很常用. 比如我们鼠标按下画线.的时候. 鼠标会移动到窗口外边.那这样的话画线就不会成功了.

实现思路:

  1.鼠标点击.保存一下坐标

  2.鼠标抬起. 创建CClientDc. 使用MoveTo移动到保存的鼠标位置.

  3.使用dc画线.画出新的位置.

例如:

鼠标按下.

void CCDCDlg::OnLButtonDown(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值

   //保存当前位置.
    m_Point = point;
}

m_point是在类中定义的一个成员.

鼠标抬起
void CCDCDlg::OnLButtonUp(UINT nFlags, CPoint point)
{
   //
    CClientDC dc(this);
    dc.MoveTo(m_Point);
    dc.LineTo(point);
}

实现截图

遇到的问题.当我们鼠标拖动到窗口外面之后.就不会画线了.所以我们要捕获鼠标.

用到的三个API:

  1.SetCapture(); 捕获鼠标.在鼠标按下的时候.

  2.GetCapture(); 获取捕获的鼠标.

  3.ReleaseCapture(); 释放

代码如下.

void CCDCDlg::OnLButtonDown(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息处理程序代码和/或调用默认值

   //保存当前位置.
    m_Point = point;
    SetCapture(); //捕获鼠标
}

void CCDCDlg::OnLButtonUp(UINT nFlags, CPoint point)
{
   //
    CClientDC dc(this);
    dc.MoveTo(m_Point);
    dc.LineTo(point);
    if (GetCapture() == this)
    {
        ReleaseCapture();
    }
}

这样我们的鼠标移动到客户区外面也可以画线了.

五丶了解键盘消息中的各位代表的含义

虚拟键代码 VK_F1... . VK开头的. 头文件中有定义.有兴趣的可以看一下.

nflgs中的第 14位 为如果先前按下.则为1.否则则为零. 其余的了解即可.

WM_KEYDOWN WM_KEYUP 可以判断键盘是否按下抬起. 通过nChar 可以确定是哪一个键.

六丶API 判断虚拟键码是否按下.

Windows 为我们提供了一个API.用来检索按键是否被按下.

GetKeyState(VK_F2)  

如果返回值为负数. 那么代表了按下. 如果返回值为0.则代表没有按.

所以我们判断返回值 < 0 则是按下 >0 则是没有按下.