当前位置:C++技术网 > 精选软件 > win32鼠标移动时的实时提示框的代码实现

win32鼠标移动时的实时提示框的代码实现

更新时间:2015-12-20 19:27:21浏览次数:1+次

    在软件很多地方都会需要这样的提示,即鼠标在客户区中移动,根据移动到的位置,提示这个位置代表的意思,比如地图、速度曲线。当移到一个一个区域的时候,提示这个区域属于哪个省,而且要实时显示,提示框要紧跟着鼠标。而速度曲线则是将鼠标移到曲线的哪里就显示这个位置所代表的速度信息。

    这些都是很实用的提示。我们这里实现鼠标移动时实时提示当前光标所在的客户区坐标值。效果图如下:

实时提示光标的坐标值的提示框

【实时提示光标的坐标值的提示框】
    从图中我们可以看到,客户区是很多的小格子,这个就是坐标线,按照每10个单位刻度画一条线。当然,我们提示坐标,并不需要依赖这个格子,在哪个位置都可以知道坐标,这样用来装逼,显得高大上点,哈哈哈。其实就是用两个循环划线就可以了。这个在WM_PAINT消息中绘制即可,代码如下:
GetClientRect(hwnd,&rect);
SelectObject(hdc,CreatePen(PS_SOLID,1,RGB(120,120,120)));
for (int i=0;i<rect.right;i++)
{
    MoveToEx(hdc,i*10,0,NULL);
    LineTo(hdc,i*10,rect.bottom);
}
for (int i=0;i<rect.bottom;i++)
{
    MoveToEx(hdc,0,i*10,NULL);
    LineTo(hdc,rect.right,i*10);
}

    代码很简单,就不详细解释了。然后我们鼠标在客户区移动时就可以看到提示框。所以我们要处理WM_MOUSEMOVE消息,在这个消息实时处理移动时提示当前光标的坐标。
    那么我们用什么作为显示提示文字的载体呢?有点经验的就知道用窗口咯。我创建一个无边框的扁平的小窗口即可,这就要用到CreateWindow函数。然后在移动光标时,我们只需要移动窗口即可跟着光标跑了。移动窗口就要用MoveWindow函数。然后要显示窗口中的文字,我们使用SetWindowText就可以设置窗口的文字。函数的使用请查MSDN。
    在移动光标也会,你可能会发现提示文字的小窗口上面,竟然有一个虚框,显得很丑。出现这个问题,是因为客户区的焦点丢失了,跑到了提示窗口上了。
    所以,我们处理客户区的焦点丢失消息WM_KILLFOCUS,在丢失的时候,又重新设置焦点到客户区,这样提示窗口就始终不会得到焦点,所以就不会出现虚框了。
    实现的完整代码如下:
#include "windows.h"
#include <Windowsx.h>//GET_X_LPARAM,GET_Y_LPARAM宏需要这个头文件
#include <tchar.h>
// - 项目是Unicode字符集
HINSTANCE g_hInstance=NULL;
LRESULT CALLBACK WinProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{
    HDC hdc;
    PAINTSTRUCT ps;
    TCHAR Info[100]=_T("C++技术网http://www.cjjjs.com");
    RECT rect;
    static HWND hTip=NULL;
    switch (message)
    {
     case WM_PAINT:
         hdc = BeginPaint(hwnd,&ps);
GetClientRect(hwnd,&rect);
SelectObject(hdc,CreatePen(PS_SOLID,1,RGB(120,120,120)));
for (int i=0;i<rect.right;i++)
{
    MoveToEx(hdc,i*10,0,NULL);
    LineTo(hdc,i*10,rect.bottom);
}
for (int i=0;i<rect.bottom;i++)
{
    MoveToEx(hdc,0,i*10,NULL);
    LineTo(hdc,rect.right,i*10);
}
         EndPaint(hwnd,&ps);
         return 0;
     case WM_KILLFOCUS:
         SetFocus(hwnd);
         return 0;
     case WM_MOUSEMOVE:
         if (!hTip)
         {
            hTip = CreateWindow(_T("Button"),_T("测试"),BS_FLAT|WS_POPUP,GET_X_LPARAM(lParam),GET_Y_LPARAM(lParam)+50,100,40,hwnd,NULL,g_hInstance,NULL);
            ShowWindow(hTip,SW_SHOWNORMAL);
         }
         else
         {
             MoveWindow(hTip,GET_X_LPARAM(lParam),GET_Y_LPARAM(lParam)+50,100,40,TRUE);
             TCHAR msg[100]=_T("");
             wsprintf(msg,_T("(%d ,%d)"),GET_X_LPARAM(lParam),GET_Y_LPARAM(lParam));
             SetWindowText(hTip,msg);
         }
         
         return 0;
     case WM_DESTROY:
         PostQuitMessage(0);
         return 0;
     default:
         return DefWindowProc(hwnd, message, wParam, lParam);
    }
}

int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrev,LPSTR lpCmd,int iShow)
{
    TCHAR ClassName[] = _T("MyClass");
    TCHAR title1[] = _T("C++技术网http://www.cjjjs.com");
    WNDCLASS wndClass;
    wndClass.cbClsExtra=0;
    wndClass.cbWndExtra=0;
    wndClass.hbrBackground= (HBRUSH)GetStockObject(WHITE_BRUSH);
    wndClass.hCursor=LoadCursor(NULL,IDC_ARROW);
    wndClass.hIcon=LoadIcon(NULL,IDI_APPLICATION);
    wndClass.hInstance = hInstance;
    wndClass.lpfnWndProc = WinProc;
    wndClass.lpszClassName = ClassName;
    wndClass.lpszMenuName=NULL;
    wndClass.style=CS_HREDRAW|CS_VREDRAW;

    if(!RegisterClass(&wndClass))return 0;
    g_hInstance = hInstance;
    HWND hwnd = CreateWindow(ClassName,title1,WS_OVERLAPPEDWINDOW,0,0,440,400,NULL,NULL,hInstance,NULL);
    ShowWindow(hwnd,SW_SHOWNORMAL);

    MSG msg;
    while (GetMessage(&msg,NULL,0,0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
    return msg.wParam;
}