1. 概述

  • MFC程序由CWinApp、MainFrm(含Menu,可用CSplitterWndEx分割)、众多Dialog等组成。
  • MFC既可以使用纯Dialog的形式,也可以使用Document+View的形式进行开发
    • 纯Dialog
    • Document+View
      • 类似于MVVM的模式
      • 可视化创建了FormView形式的Dialog后,可以使用Project->Class Wizard创建这个FormView相应的ModelView类
  • MFC应用程序的入口函数(初始化函数)
    • BOOL XXXApp::InitInstance()
    • MFC对WindowsAPI进行了封装。MFC提供给编程人员的第一个裸露程序入口就是CWinApp的InitInstance(),其实最终的入口函数还是WindowsAPI的WinMain()函数
    • 所以在自定义了主界面的Init()等函数时,通常就是在InitInstance()中调用它

2. 通过代码控制界面

  • 并没有主界面的可视化设计界面?只能通过Resource View下的Menu去修改Menu??
  • 通过代码中的CSplitterWndEx等进行界面的切割
    • 在头文件MainFrm.h,中定义分割类对象和相关初始化函数(HeaderView是生成的Class的名字):
// Overrides
public:
    virtual BOOL OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext);
protected:  // control bar embedded members
    CSplitterWndEx   m_wndSplitterHoriz1;
private:
    BOOL CreateSplitterWindow(CCreateContext* pContext);
* 在MainFrm.cpp中给出splitter及view的设置:
    BOOL CMainFrame::Init()
    {
        // set default
        INT iYCur  = 300;
        INT iYMin  = 0;

        m_wndSplitterHoriz1.SetRowInfo(0, iYCur, iYMin);
        m_wndSplitterHoriz1.RecalcLayout();

        return TRUE;
    }

    BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT /*lpcs*/,
        CCreateContext* pContext)
    {
        //SetAMMWindowTitle(); // no project

        BOOL bRet = CreateSplitterWindow(pContext);
        return bRet;
    }

    // -----------------------------------------------------------
    // function builds the splitter and the own views together
    BOOL CMainFrame::CreateSplitterWindow(CCreateContext* pContext)
    {
        //将主窗口分割为2行1列,
        if (!m_wndSplitterHoriz1.CreateStatic(this, 2, 1))
        {
            return FALSE;
        }
        //然后在主窗口的第2行第1列中绑定view
        m_wndSplitterHoriz1.CreateView(0,0,RUNTIME_CLASS(HeaderView),CSize(100,100),pContext);
        m_wndSplitterHoriz1.SetRowInfo(0,200,0);
        m_wndSplitterHoriz1.CreateView(1,0,RUNTIME_CLASS(HeaderView),CSize(100,100),pContext);
        m_wndSplitterHoriz1.SetRowInfo(1,200,0);

        return TRUE;
    }
* 设置切割布局时需要制定分割线两侧是什么View(MFC的文档,是一种特殊的Dialog,在Resource View中的Dialog文件夹右击选择Add Resource时,弹出窗的Dialog节点可以展开,下面有一个FormView节点,用于创建这种View,并且可以在属性窗口修改ID)

3. 可视化设计界面

  • 可在Resource View看到所有的界面
  • 主界面的菜单
    • 主界面的菜单UI是XXX.rc/Menu/IDR_MAINFRAME,可以直接可视化的增删改菜单项,设置文本(用@标识用于多语言替换?)、ID、TEXT等。
    • 在主界面MainFrm.cpp中可以用菜单的ID来进行消息映射绑定,消息绑定的事件中也可以Qt的界面
  • 右键菜单
    • 可以在XXX.rc/Menu/IDR_CONTEXT_MENU中进行可视化设置,可以设置ID如IDR_CONTEXT_MENU
    • 可以包含多个右键菜单,有各自的ID如IDX_CONTEXT_MENU_LOCAL_LIST_VIEW,代码中动态获得应该显示哪一个右键菜单
    UINT uiRessourceIDContextMenus = IDR_CONTEXT_MENU; 

    int iIdxContextMenu = -1; // -1 = undefined

    switch (contextMenu)
    {
    case ContextMenuLocalListView:
        iIdxContextMenu = IDX_CONTEXT_MENU_LOCAL_LIST_VIEW;
        break;
    case ContextMenuLocalTreeView:
        iIdxContextMenu = IDX_CONTEXT_MENU_LOCAL_TREE_VIEW;
        break;
    case ContextMenuRemoteListView:
        iIdxContextMenu = IDX_CONTEXT_MENU_REMOTE_LIST_VIEW;
        break;
    case ContextMenuRemoteTreeView:
        iIdxContextMenu = IDX_CONTEXT_MENU_REMOTE_TREE_VIEW;
        break;
    case ContextMenuEmpty:
        iIdxContextMenu = IDX_CONTEXT_MENU_EMPTY;
        break;
    case ContextMenuOutputWindow:
        iIdxContextMenu = IDX_CONTEXT_MENU_OUTPUT_VIEW;
        break;
    default:
        assert(false); // implementation error -> undefined context menu
        PRINT_TRACE(AMMCOMP_TRACE_LIB_DIALOGS, AMMCOMP_TRACE_LAYER_GUI, AMMCOMP_TRACE_LEVEL_ERROR, FUNCTION_SIGNATURE + " undefined context menu");
        break;
    }

    CMenu oContextMenus;
    if (FALSE != oContextMenus.LoadMenu(uiRessourceIDContextMenus)) // contains all context menus
    {
        CMenu* pPopup = oContextMenus.GetSubMenu(iIdxContextMenu);  // get the wanted context menu
        if(NULL != pPopup)
        {
            if (NULL != pWndParent)
            {
                while (pWndParent->GetStyle() & WS_CHILD)
                {
                    if (NULL != pWndParent)
                    {
                        pWndParent = pWndParent->GetParent();
                    }
                }
            }

            if (NULL != pWndParent)
            {
                pPopup->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point.x, point.y, pWndParent);
            }
        }
    }
  • 对话框

    • 所有的Dialog都在XXX.rc/Dialog/下面
    • 在可视化设计界面中的某个控件上右击可以增加:选择"Add Variable..."
    • 添加、设计Dialog
      • 使用PROJECT->Class Wizard...按步骤设计生成MFC界面

        • 点击"Add Class..."按钮,在弹窗中输入想要创建的窗口名字(也就是Class name,.h和.cpp文件会自动同步)
      • 也可以直接在Resource View的Dialog等文件夹上直接右击添加
        • 保存后会直接在_header/resource.h文件中生成相应的宏、常量,在.rc文件中生成界面相关代码
      • 绑定Dialog弹出的消息映射
        • 菜单点击触发:
        • 页面上的按钮点击触发:
      • 可以使用Toolbox进行界面设计

4. 数据及事件绑定(通过Variable)

  • 数据绑定(MFC中叫数据交换和检验/校验,winform中直接通过控件获取text什么的,相当于已经帮你绑好了)

    • 代码中通过DoDataExchange进行数据双向绑定:
    void CDlgTabMaintenance::DoDataExchange(CDataExchange* pDX)
    {
      CRCSBasicProjectDialogTab::DoDataExchange(pDX);
      DDX_Control(pDX, IDC_TREE_FILES_MAINTENANCE, m_ctrlTreeFiles);
    }
    • 可视化操作中,右键控件选择"Add Variable...",同样会在cpp和头文件中生成DoDataExchange部分代码,将控件和变量绑定在一起,也会在.rc和Resource.h文件中生成对应的常量等

      • 如果勾选"Control Variable",那么添加变量为控件变量,可以操作控件,变量类型将固定为控件类型;如果不勾选,则让控件关联了一个变量,不可以修改控件
      • 将在头文件中声明相应的变量。
      • 同时如果是控件变量,则会在cpp文件中生成DoDataExchange代码段;如果不是控件变量,那么将在.cpp文件的类定义的继承后面多一段, edit_binded_data(0)。
      • 完成变量绑定后,就不需要管它了,在文本变化等事件的处理函数中,会自动把变量值变成控件中用户填的新值
  • 在Class中通过消息映射表进行事件绑定
    • 可以手动写事件处理函数和向消息映射表添加记录
    • 可以在设计窗口右击控件选择"Add Event Handler..."
    • 可以在控件的属性窗口中选择事件tab,然后在想要加的事件右侧填函数名
  • 打开一个新的MFC Dialog可以用CreateProcess,比如按钮点击触发或菜单按钮点击触发

5. 其他

  • MainFrm.cpp中的Init()等函数在定义前,都需要现在.h头文件中声明函数原型,否则无法识别报错
  • 可以在函数中任意位置临时#include?? -- 不行,即使vs中可以找到,编译也可能不通过,可能要写到函数中的话,需要特殊定义方法或者当成了内部类?
  • 如果Resource.h或者.rc文件被打开了,那么Resource View不能展开,同时这两个文件和设计窗口不能被同时打开
  • 有时Resource View下面节点无法展开,可以先把编辑窗口所有打开的文件关掉再试,可能是rc文件被打开等问题?
  • 给主界面的菜单项绑定的事件,必须声明、定义和绑定分别要放在MainFrm.h和MainFrm.cpp中,否则运行后菜单会一直是灰色,并不是enable的原因
    • Resource View中给菜单项添加Event Handler时COMMAND的Class List中要选CMainFrame?
  • 报错"unexpected #endif"或者"unexpected end of file while looking for precompiled header. Did you forget to add '#include "stdafx.h"' to your source?"
    • 通常是因为微软忽略掉实现文件中stdafx.h包含头文件之前的一切指令
    • 所谓头文件预编译,就是把一个工程(Project)中使用的一些MFC标准头文件(如Windows.H、Afxwin.H)预先编译,以后该工程编译时,不再编译这部分头文件,仅仅使用预编译的结果。这样可以加快编译速度,节省时间。
    • 编译器通过一个头文件stdafx.h来使用预编译头文件。stdafx.h这个头文件名是可以在project的编译设置里指定的。编译器认为,所有在指令#include "stdafx.h"前的代码都是预编译的,它跳过#include "stdafx. h"指令,使用projectname.pch编译这条指令之后的所有代码。
    • 解决
      • #include "stdafx.h"要放在最前面
      • 不使用预编译头文件(stdafx.h)
      • 在stdafx.h文件中增加预编译处理
      • 还有一种说法是:project_name[right click] -> C/C++ -> Precompiled Header -> Precompiled Header (Choose Not Using Precompiled Headers)
    • 参考参考2

MFC学习(三):项目学习的更多相关文章

  1. java_web学习(三) eclipse_jsp学习

    1.首先打开eclipse,新建一个Dynamac web project项目文件 2.在WebContent单击右键创建JSP File 3.过程 4.简单的jsp代码 运行结果: 5.导出war文 ...

  2. Html学习(三) 分类学习

    代码: <h1>这是一级分类吗</h1> <h2>这是二级分类吗</h2> <h3>这是三级分类吗 </h3> 效果: 介绍: ...

  3. 中小研发团队架构实践之生产环境诊断工具WinDbg 三分钟学会.NET微服务之Polly 使用.Net Core+IView+Vue集成上传图片功能 Fiddler原理~知多少? ABP框架(asp.net core 2.X+Vue)模板项目学习之路(一) C#程序中设置全局代理(Global Proxy) WCF 4.0 使用说明 如何在IIS上发布,并能正常访问

    中小研发团队架构实践之生产环境诊断工具WinDbg 生产环境偶尔会出现一些异常问题,WinDbg或GDB是解决此类问题的利器.调试工具WinDbg如同医生的听诊器,是系统生病时做问题诊断的逆向分析工具 ...

  4. Spring Boot 项目学习 (三) Spring Boot + Redis 搭建

    0 引言 本文主要介绍 Spring Boot 中 Redis 的配置和基本使用. 1 配置 Redis 1. 修改pom.xml,添加Redis依赖 <!-- Spring Boot Redi ...

  5. 20145337《Java程序设计》第三周学习总结

    20145337 <Java程序设计>第三周学习总结 教材学习内容总结 类与对象 类与对象的关系:要产生对象必须先定义类,类是对象的设计图,对象是类的实例.我觉得在视频中对类与对象关系的描 ...

  6. 转: 学习开源项目的若干建议(infoq)

    转: http://www.infoq.com/cn/news/2014/04/learn-open-source 学习开源项目的若干建议 作者 崔康 发布于 2014年4月11日 | 注意:GTLC ...

  7. 华为章宇:如何学习开源项目及Ceph的浅析

    转自http://www.csdn.net/article/2014-04-10/2819247-how-to-learn-opensouce-project-&-ceph 摘要:开源技术的学 ...

  8. DjangoRestFramework学习三之认证组件、权限组件、频率组件、url注册器、响应器、分页组件

    DjangoRestFramework学习三之认证组件.权限组件.频率组件.url注册器.响应器.分页组件   本节目录 一 认证组件 二 权限组件 三 频率组件 四 URL注册器 五 响应器 六 分 ...

  9. 20175227张雪莹 2018-2019-2 《Java程序设计》第三周学习总结

    20175227张雪莹 2018-2019-2 <Java程序设计>第三周学习总结 教材学习内容总结 (仅在此列举个性化学习总结) 一.编程语言的几个发展阶段. 1.面向机器语言:汇编语言 ...

  10. 20175314 《Java程序设计》第三周学习总结

    20175314 <Java程序设计>第三周学习总结 教材学习内容总结 编程语言的发展事是从面向机器(汇编.机器)到面向过程(C)再到面向对象(Java) 成员变量: 1.成员变量定义在类 ...

随机推荐

  1. 【源码阅读】Java集合之三 - ArrayDeque源码深度解读

    Java 源码阅读的第一步是Collection框架源码,这也是面试基础中的基础: 针对Collection的源码阅读写一个系列的文章,本文是第三篇ArrayDeque. ---@pdai JDK版本 ...

  2. IDEA debug

    版权声明: 本文转自:https://blog.csdn.net/qq_27093465/article/details/64124330 1,rerun XXX,这个就是直接重新跑某个程序.2,这个 ...

  3. C# StopWatch的BUG????

    //BUG?????? //使用StopWatch测试运行时间 //两段测试A和B //测试结果受测试顺序影响,后测要比先测耗时长了许多 static void TestKeyIntStr() { v ...

  4. 关于springboot中文件上传,properties配置

    spring.http.multipart.enabled=true #默认支持文件上传. spring.http.multipart.file-size-threshold=0 #支持文件写入磁盘. ...

  5. 数据库,使用Druid 加密数据库密码

    首先我们得下载一个druid-1.0.16.jar的包 其次键入命令 java -cp druid-1.0.16.jar com.alibaba.druid.filter.config.ConfigT ...

  6. android学习-Eclipse中修改Android项目图标

    参考原文:http://blog.csdn.net/wpwbb510582246/article/details/52556753 方法一:替换res文件夹下的ic_launcher-web.png图 ...

  7. 结队编程第二次作业:Android自动生成算式应用

    一.题目要求 本次作业要求两个人合作完成,驾驶员和导航员角色自定,鼓励大家在工作期间角色随时互换,这里会布置两个题目,请各组成员根据自己的爱好任选一题. 这次我和我的小伙伴选择了题目一. 题目1: 实 ...

  8. 字符图元 & 显示列表

    [字符图元] 1.typeface(字样),即设计风格,如Courier等. 2.font(字体),如10磅Courier斜体. 3.monspace即为等宽字体,proportional为非等宽字体 ...

  9. 对于Android NDK编译器ARM和Thumb模式的理解

    编译NDK项目时,编译器无法识别arm汇编,设置LOCAL_ARM_MODE := arm后问题解决, NDK文档上对LOCAL_ARM_MODE的说明如下: LOCAL_ARM_MODE By de ...

  10. [erlang 002]gen_server中何时会跑到terminate函数

    一.从start方法产出的独立gen_server进程 实验代码: %%%-------------------------------------- %%% @Module  : %%% @Auth ...