// winsdk.cpp : 定义应用程序的入口点。
//#include "stdafx.h"
#include "winsdk.h"
#include <commctrl.h>          //必须导入此头文件
#define MAX_LOADSTRING 100// 全局变量:
HINSTANCE hInst; // 当前实例
TCHAR szTitle[MAX_LOADSTRING]; // 标题栏文本
TCHAR szWindowClass[MAX_LOADSTRING]; // 主窗口类名HWND m_hWnd1;//客户区中的按钮// 此代码模块中包含的函数的前向声明:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM);int APIENTRY _tWinMain(HINSTANCE hInstance,
   HINSTANCE hPrevInstance,
   LPTSTR    lpCmdLine,
   int       nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine); // TODO: 在此放置代码。
MSG msg;
HACCEL hAccelTable; // 初始化全局字符串
LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadString(hInstance, IDC_WINSDK, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance); // 执行应用程序初始化:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
} hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_WINSDK)); // 主消息循环:
while (GetMessage(&msg, NULL, 0, 0))
{
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
} return (int) msg.wParam;
}//
//  函数: MyRegisterClass()
//
//  目的: 注册窗口类。
//
//  注释:
//
//    仅当希望
//    此代码与添加到 Windows 95 中的“RegisterClassEx”
//    函数之前的 Win32 系统兼容时,才需要此函数及其用法。调用此函数十分重要,
//    这样应用程序就可以获得关联的
//    “格式正确的”小图标。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEX wcex; wcex.cbSize = sizeof(WNDCLASSEX); wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_WINSDK));
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = MAKEINTRESOURCE(IDC_WINSDK);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL)); return RegisterClassEx(&wcex);
}//
//   函数: InitInstance(HINSTANCE, int)
//
//   目的: 保存实例句柄并创建主窗口
//
//   注释:
//
//        在此函数中,我们在全局变量中保存实例句柄并
//        创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
HWND hWnd; hInst = hInstance; // 将实例句柄存储在全局变量中 hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); if (!hWnd)
{
return FALSE;
} ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd);
//加个ListBox,想测试鼠标移过事件  m_hWnd1=CreateWindow(L"LISTBOX", NULL, 
WS_CHILD | WS_VISIBLE | WS_BORDER ,
3,25,245,654, 
0, 
(HMENU)1234, 
NULL, NULL ); return TRUE;
}//
//  函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
//  目的: 处理主窗口的消息。
//
//  WM_COMMAND - 处理应用程序菜单
//  WM_PAINT - 绘制主窗口
//  WM_DESTROY - 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
int wmId, wmEvent;
PAINTSTRUCT ps;
HDC hdc;
switch (message)
{
case WM_COMMAND:
wmId    = LOWORD(wParam);
wmEvent = HIWORD(wParam);
// 分析菜单选择:
switch (wmId)
{
case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
break;
case WM_PAINT:
hdc = BeginPaint(hWnd, &ps);
// TODO: 在此添加任意绘图代码...
EndPaint(hWnd, &ps);
break; case WM_MOUSELEAVE: 
MessageBox(hWnd,TEXT("Mouse离开!"),TEXT("MouseLeave"),0); 
break;
case WM_MOUSEHOVER: 
MessageBox(hWnd,TEXT("Mouse通过"),TEXT("MouseHover"),0); 
break; 
case WM_NCMOUSELEAVE: 
//MessageBox(hWnd,TEXT("MouseNClient!"),TEXT("MouseNClient!"),0); 
break;
case WM_MOUSEMOVE:    
  TRACKMOUSEEVENT trmouse;  
  trmouse.cbSize = sizeof(TRACKMOUSEEVENT); 
  trmouse.dwFlags = TME_LEAVE | TME_HOVER ;
  trmouse.dwHoverTime = HOVER_DEFAULT; 
  trmouse.hwndTrack = m_hWnd1;  //为什么此处设置后不起效果,用hWnd的话,鼠标移过ListBox不起作用,在其它区起作用
  if(!_TrackMouseEvent(&trmouse))  
  return FALSE;
  
  break; case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}// “关于”框的消息处理程序。
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE; case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}

解决方案 »

  1.   

      trmouse.hwndTrack = m_hWnd1;  //为什么此处设置后不起效果,用hWnd的话,鼠标移过ListBox不起作用,在其它区起作用 
      

  2.   

    你要在某个窗口中响应消息,就应该子类化该窗口,比如你的ListBox。你的设置trmouse.hwndTrack = m_hWnd1是起作用的,但是你在父窗口响应WM_MOUSELEAVE当然没有作用。
      

  3.   

    感谢2楼的帮忙,我这是在一个win32项目里做的,没有用mfc,按钮可以给父窗口发WM_COMMAND消息,并在父窗口的消息处理函数中处理,为什么不能处理WM_MOUSELEAVE消息呢?能不能给个实现代码?
      

  4.   

    那你知道你的鼠标经过ListBox时,ListBox的窗口过程的会处理WM_MOUSEMOVE消息之后。它会产生什么消息送给你的主窗口过程。你应该子类化ListBox窗口。
    WNDPROC wpOrigListBoxProc; 
     
    LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) 

        HWND hListBox; 
     
        switch(uMsg) 
        { 
            case WM_CREATE: 
                // Retrieve the handle to the edit control. 
                hwndListBox = GetDlgItem(hWnd, ID_ListBox); 
     
                // Subclass the edit control. 
                wpOrigListBoxProc = (WNDPROC) SetWindowLong(hwndListBox, 
                    GWL_WNDPROC, (LONG) ListBoxSubclassProc); 
                // 
                // Continue the initialization procedure. 
                // 
                return TRUE; 
     
            case WM_DESTROY: 
                // Remove the subclass from the edit control. 
                SetWindowLong(hwndListBox, GWL_WNDPROC, 
                    (LONG) wpOrigListBoxProc); 
                // 
                // Continue the cleanup procedure. 
                // 
                break; 
        } 
        return FALSE; 
            UNREFERENCED_PARAMETER(lParam); 

     
    // Subclass procedure 
    LRESULT APIENTRY ListBoxSubclassProc(
        HWND hwnd, 
        UINT uMsg, 
        WPARAM wParam, 
        LPARAM lParam) 

        if (uMsg == WM_MOUSEMOVE) 
            SENDmessage(   );//向主窗口发消息
     
        return CallWindowProc(wpOrigListBoxProc, hwnd, uMsg, 
            wParam, lParam); 

    自己参考一下吧
      

  5.   

    那你知道你的鼠标经过ListBox时,ListBox的窗口过程的会处理WM_MOUSEMOVE消息之后。它会产生什么消息送给你的主窗口过程。你应该子类化ListBox窗口。
    WNDPROC wpOrigListBoxProc; 
     
    LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) 

        HWND hListBox; 
     
        switch(uMsg) 
        { 
            case WM_CREATE: 
                // Retrieve the handle to the edit control. 
                hwndListBox = GetDlgItem(hWnd, ID_ListBox); 
     
                // Subclass the edit control. 
                wpOrigListBoxProc = (WNDPROC) SetWindowLong(hwndListBox, 
                    GWL_WNDPROC, (LONG) ListBoxSubclassProc); 
                // 
                // Continue the initialization procedure. 
                // 
                return TRUE; 
     
            case WM_DESTROY: 
                // Remove the subclass from the edit control. 
                SetWindowLong(hwndListBox, GWL_WNDPROC, 
                    (LONG) wpOrigListBoxProc); 
                // 
                // Continue the cleanup procedure. 
                // 
                break; 
        } 
        return FALSE; 
            UNREFERENCED_PARAMETER(lParam); 

     
    // Subclass procedure 
    LRESULT APIENTRY ListBoxSubclassProc(
        HWND hwnd, 
        UINT uMsg, 
        WPARAM wParam, 
        LPARAM lParam) 

        if (uMsg == WM_MOUSEMOVE) 
            SENDmessage(   );//向主窗口发消息
     
        return CallWindowProc(wpOrigListBoxProc, hwnd, uMsg, 
            wParam, lParam); 

    自己参考一下吧
      

  6.   

    果然是高手,我知道有这个思路,但是不知怎么实现,我也是这方面的新手, 
      GetDlgItem(hWnd, ID_ListBox);  SetWindowLong()没用过,感谢帮助,这真是个好地方。