`
lunan
  • 浏览: 77995 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

MFC 动态菜单 及响应(转)

阅读更多
做一个MFC程序的时候碰到一个需求。就是需要根据定制情况,动态生成菜单,菜单的具体结构和信息是之前不知道的(因此不能利用工具构造),点击不同类型的菜单会触发特定的一类事件(需要动态绑定事件)。这种需求实际是蛮不BT的,很多场合下都可能会有,用C#写了个Demo花了不到半个小时。但转到MFC下来写,就费尽周折。其实这个问题代表了在MFC中动态创建资源绑定事件的一般性问题,所以总结一下。

动态创建菜单需要先了解CMenu类。通常我们利用工具绘制一个菜单,每一个菜单项下都可以视为有一个CMenu类。它们联系在一起,形成树状。典型的一个菜单对应过来是如下图这个样子:




如上,CMenu可以分成三种,一个是Popup(黄色),一个是Separator(灰色),一个是Item(红色)。前两种都是没有ID信息的,Popup有一个指针,指向其SubMenu;Item保存各种信息有ID可以响应事件;Separator,恩,基本是一穷二白的。

CMenu的CreateMenu方法可以创建一个菜单资源,用DeleteMenu(包含所有子菜单)或DestoryMenu可以销毁菜单资源,用AppendMenu可以添加一个菜单。了解这些内容,就可以开工了,现实现上图所示的MainSubMenu1下菜单的动态创建,代码如下:

    // 假设在ChildFrm中,调用该方法获得当前的主菜单指针
    CMenu* mainMenu = AfxGetMainWnd()->GetMenu();
    CMenu* subMenu = NULL;



    // 遍历主菜单下的各级菜单寻找名为MainSubMenu1的菜单

    int menuCount = mainMenu->GetMenuItemCount();
    for(int i = 0; i < menuCount; i++)
    {
        CString menuName;
        if(mainMenu->GetMenuStringA(i, menuName, MF_BYPOSITION)
            && menuName == "&MainSubMenu1")
        {
            drawingMenu = mainMenu->GetSubMenu(i);
            break;
        }
    }

    // 移除原有的菜单项
    int subMenu1Count = subMenu->GetMenuItemCount();
    for(int i = subMenu1Count - 1; i >= 0; i--)
    {
        subMenu->DeleteMenu(i, MF_BYPOSITION);
    }

    // 动态添加Item菜单项
    for(int i = 0; i < 2; i++)
    {

        CString message = "";

        subMenu->AppendMenuA(MF_STRING, ID_BEGIN + i, message.Format("SubSubMenu%i", i);

    }


    // 添加分隔符

    subMenu->AppendMenuA(MF_SEPARATOR);


    // 添加弹出式子菜单

    CMenu * popupMenu = new CMenu();
    popupMenu->CreateMenu();
    for(int i = 0; i < 2; i++)
    {

        CString message = "";

        popupMenu->AppendMenuA(MF_STRING, ID_BEGIN + 2 + i, message.Format("PopupSubMenu%i", i));

    }
    subMenu->AppendMenuA(MF_POPUP, (UINT_PTR)popupMenu->operator HMENU(), "PopupMenu");

有几个需要注意的地方,一个是主菜单的指针获得,可以参考《MFC框架各部分指针获取方式》一文。另一个是Popup的菜单建立,策略是分成两部分,先new出内存在Create出资源,缺一不可。最后一个是为每个Item菜单合理分配ID,这些ID须事先预留出来,在MFC中,至少40000到49000通常都是没人用。

这也就引出下一个问题,即菜单事件的动态绑定。我们知道在.net中,事件是真正动态绑定的,而MFC中的事件都是只能静态绑定,这是由两者的编译方式决定的。所以,在MFC中需要定义菜单事件,你需要先挖好坑(预留足够ID),规定每个坑种什么罗卜(将不同类型的ID绑定到不同类别的事件处理函数上),最后才能按坑种罗卜(为执行相应事件的菜单设置相应的ID)。

可以有两种方式来绑定对应ID处理的事件,一个是通过ON_COMMAND_RANGE宏(想一下ON_COMMAND宏会不会派上用场?)在MessageMap里绑定批量处理事件的函数;另一个是重载PreTranslateMessage函数,截获并判断ID来进行处理。思想都是类似的。值得注意的是,通常还需要配套使用ON_UPDATE_COMMAND_UI_RANGE来保证动态创建的菜单Enable为True,否则很可能菜单不可以点击
-------------------------------------------------


重载PreTranslateMessage
BOOL   C*****View::PreTranslateMessage(MSG*   pMsg)  
{
//   TODO:   Add   your   specialized   code   here   and/or   call   the   base   class
if(pMsg-> message==WM_RBUTTONDOWN)
if(pMsg-> hwnd==this-> GetListCtrl().m_hWnd)
{
                        CMenu   menu;
        CPoint   xy;
                        ::GetCursorPos(&xy);
        menu.CreatePopupMenu();//添加菜单
                        menu.AppendMenu(MF_STRING,IDM_SHOW, "显示所有 ");
        menu.AppendMenu(MF_STRING,IDM_ADD, "增加记录 ");
        int   count=this-> GetListCtrl().GetSelectedCount();
        if(count==1)
                menu.AppendMenu(MF_STRING,IDM_MODIFY, "修改记录 ");
        else
                                menu.AppendMenu(MF_STRING|MF_GRAYED,IDM_MODIFY, "修改记录 ");
                        if(count> 0)
                menu.AppendMenu(MF_STRING,IDM_DELETE, "删除记录 ");
        else
                                menu.AppendMenu(MF_STRING|MF_GRAYED,IDM_DELETE, "删除记录 ");
                        menu.AppendMenu(MF_STRING,IDM_FIND, "查找记录 ");
                        menu.AppendMenu(MF_STRING,IDM_MANAGE, "管理 ");
        menu.TrackPopupMenu(TPM_LEFTALIGN|TPM_LEFTBUTTON,xy.x,xy.y,this);
                        return   TRUE;
}
return   CListView::PreTranslateMessage(pMsg);
}

---------------------------------------------------------

MFC中利用CMenu类动态添加弹出菜单和响应函数

步骤:

1 声明一个菜单:

    CMenu menu;

2 生成菜单对象: 

    menu.CreatePopupMenu();

3 给菜单添加上内容:

   menu.AppendMenu(MF_STRING,WM_CLEARHOSTS,"清除HOSTS");

AppendMenu函数具体的意义可以查看MSDN,其中WM_CLEARHOSTS为自定义的一个消息,最后一个参数为菜单的text,点击这个菜单就可以调用WM_CLEARHOSTS消息的处理函数。

4 添加子弹出菜单:

    CMenu submenu;

    submenu.CreatePopupMenu();

    menu.AppendMenu(MF_POPUP,(UINT_PTR)(submenu.m_hMenu),"sub");

5 设置当失去焦点时菜单自动消失

    SetForegroundWindow();

6 设置菜单的位置:

    menu.TrackPopupMenu();

以上各步连接起来就是一个完整的动态生成菜单的步骤,当动态生成的菜单很多且菜单又不固定的时候,预先

为每个菜单都定义一个消息和消息处理函数是很麻烦且不现实的,现在介绍一种方法来动态响应动态生成的菜单。

其原理就是利用OnCommand函数。

首先,要为每一个动态生成的菜单指定一个ID,方式如下

    menu.AppendMenu(MF_STRING,ID,"yourMenuName");

其中参数ID为一个唯一的整数,可以由你自己指定,当鼠标单击此菜单的时候,系统发送一个消息,此消息优先

被OnCommand函数接收,OnCommand函数的原形为:

BOOL OnCommand(WPARAM wParam, LPARAM lParam);

假如你指定菜单的ID为10001,响应函数的具体写法为:

BOOL OnCommand(WPARAM wParam, LPARAM lParam)

{

    int menuID = LOWORD(wParam);
    if(menuID > 10000)

    {

      //添加你自己的处理代码

    }

}

如果是使用对话框的mfc,自己重载OnCommand函数即可

--------------------------------------------------------------

响应WM_NOTIFY()
答!: 2:
BOOL CTest6Dlg::PreTranslateMessage(MSG* pMsg)

{
if(pMsg->message == WM_RBUTTONDOWN)

{

if(pMsg->hwnd == GetDlgItem(IDC_LISTBOX1)->m_hWnd)

{

DWORD dwPos = GetMessagePos();

CPoint point( LOWORD(dwPos), HIWORD(dwPos) );




CMenu menu;

VERIFY( menu.LoadMenu( IDR_MENU1 ) );

CMenu* popup = menu.GetSubMenu(0);

ASSERT( popup != NULL );

popup->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point.x, point.y, this );

}

}
return CDialog::PreTranslateMessage(pMsg);

}
分享到:
评论

相关推荐

    为MFC对话框程序添加菜单和菜单响应函数

    以上步骤就是为MFC对话框程序添加菜单和处理菜单响应的基本流程。通过这种方式,我们可以扩展对话框的功能,使其不仅仅局限于对话框内的交互,而是可以响应用户的菜单操作,实现更丰富的功能。在实际开发中,你可能...

    MFC-动态添加和删除菜单栏

    在实际应用中,可能需要响应用户的某些操作来动态改变菜单。这可以通过消息映射机制实现。例如,当用户触发某个事件时,可以在OnUpdateCmdUI成员函数中更新菜单的状态: ```cpp ON_UPDATE_COMMAND_UI(ID_FILE_OPEN,...

    MFC菜单响应事件绘图

    总的来说,MFC菜单响应事件是通过消息映射机制将用户在菜单上的操作转化为代码执行的过程。在绘图应用程序中,这个过程使得我们可以根据用户的指令在窗口上绘制出各种图形,为用户提供直观的交互方式。通过理解并...

    MFC右键菜单及文件窗口测试.zip

    本示例项目"MFC右键菜单及文件窗口测试.zip"聚焦于两个关键组件:CMenu和CFileDialog,它们是MFC中处理右键菜单和文件对话框的核心类。 首先,我们来看CMenu。CMenu类是MFC中的一个抽象,它允许开发者创建、操作和...

    mfc的菜单操作

    在实际项目中,你可能会遇到更复杂的需求,比如动态添加菜单项、响应子菜单事件等,但基本原理都是围绕`CMenu`类及其成员函数展开的。通过理解和熟练掌握这些知识点,你将能够轻松地在MFC应用中实现各种丰富的菜单...

    vc mfc 创建动态菜单

    本教程将深入探讨如何在VS2008中利用MFC实现动态菜单的创建。 动态菜单意味着在程序运行时,而不是在设计阶段,根据需要创建和修改菜单项。这对于那些需要根据用户权限、程序状态或特定条件来定制菜单的软件来说,...

    c++ mfc应用程序实现浮动菜单

    当然,实际应用中可能还需要处理更多的细节,如键盘快捷键、菜单项的动态更新、菜单的动画效果等。通过熟练掌握MFC的类库和Windows消息机制,开发者可以轻松地构建出功能丰富的浮动菜单系统,提高用户界面的易用性。...

    MFC实现菜单项的编写源码

    在这里,`CMyApp`是你的应用程序类,`OnFileOpen`和`OnFileClose`是响应菜单项选择的成员函数。这些函数通常会执行与用户操作相关的具体任务。 为了使菜单可见,你需要在窗口初始化或创建时将菜单附加到窗口。这...

    VS2010基于对话框的MFC的菜单

    在本文中,我们将深入探讨如何在Visual Studio 2010中使用MFC(Microsoft Foundation Classes)库创建基于对话框的菜单。...通过实践这些示例代码,你将能更好地掌握MFC菜单系统,并能够将这些技术应用到自己的项目中。

    基于MFC+XML实现的动态菜单及工具栏创建的程序代码

    本程序代码“基于MFC+XML实现的动态菜单及工具栏创建”是将这两者结合,实现了菜单和工具栏的动态加载和管理。 1. **MFC框架理解**: MFC提供了一套面向对象的类库,它封装了Windows API,使得开发者可以使用C++的...

    vc++与MFC菜单编程学习

    MFC 程序设计菜单是 VC++ 中的一种重要组件,菜单的创建、加载、显示和响应命令是 MFC 编程的核心内容。本文将详细介绍 MFC 中使用 VC++ 创建菜单的方法,并提供实例学习,旨在帮助读者学习 MFC。 菜单分类 -------...

    MFC中的菜单基本应用

    五、自定义菜单响应 除了MFC默认的命令处理机制,还可以重写CWnd的OnMenuSelect()函数来处理自定义的菜单选择事件。在这个函数中,可以根据选择的菜单项进行特定的逻辑处理。 总结,MFC中的菜单和工具栏提供了丰富...

    mfc菜单编辑

    以下是对MFC菜单编辑的详细说明: 1. **菜单ID的使用**: MFC中,菜单ID通常是大写字母表示,如IDI_光标和IDC_菜单。这些ID用于唯一标识菜单项,方便程序内部处理菜单事件。在创建菜单项时,可以通过`IDM_[查看]`-...

    MFC编写高级弹出菜单

    在Windows编程领域,MFC(Microsoft Foundation ...在实际编程过程中,还需要考虑菜单响应事件的处理、菜单项的状态更新以及可能的国际化和本地化需求。熟练掌握这些知识点,可以提升Windows应用的用户体验和专业性。

    VC6_menu.rar_MFC menu_mfc菜单弹出_vc6.0菜单设计_动态菜单_原理图

    关于VC6.0的菜单界面设计的源代码,菜单的工作原理及编写应用,菜单命令消息在MFC框架程序的几个类中的传递顺序和处理过程。标记菜单、缺省菜单的实现原理、图形菜单的实现及常犯错误的分析,GetSystemMetrics的应用...

    VC.rar_mfc 菜单_vc 6.0mfc_vc++6.0mfc_菜单 对话框

    《VC++6.0 MFC菜单与对话框实践教程》 在编程领域,Microsoft Foundation Class (MFC) 库是C++程序员开发Windows应用程序的重要工具,尤其在VC++6.0时代,MFC的使用非常广泛。本教程将深入探讨如何使用MFC库创建和...

    MFC 基于对话框 菜单栏的选择对号标记

    总之,实现MFC对话框中的菜单栏选择对号标记,需要理解`CMenu`类和菜单项的状态管理,以及如何在对话框类中响应菜单事件。通过`CheckMenuItem()`和消息映射机制,我们可以轻松地控制菜单项的选中状态,从而为用户...

    MFC对话框右键菜单

    **MFC对话框右键菜单**是Windows应用程序开发中的一个重要概念,主要涉及到用户界面的交互设计。MFC(Microsoft Foundation Classes)是微软提供的一套面向对象的类库,用于简化在Windows平台上用C++开发应用程序的...

Global site tag (gtag.js) - Google Analytics