本人在学习MFC的过程中,发现下面这个比较好的教程,这边转载记录一下
原文地址:MFC 最详细入门教程
From:https://blog.csdn.net/wang18323834864/article/details/78621633/
Visual Studio 2019:https://visualstudio.microsoft.com/zh-hans/
鸡啄米 ----- VS2010/MFC编程入门教程之目录和总结:http://www.jizhuomi.com/software/257.html
一、VS2010/MFC编程入门教程之目录
第一部分:VS2010/MFC开发环境
VS2010/MFC编程入门之前言
VS2010/MFC编程入门之一(VS2010与MSDN安装过程图解)
第二部分:VS2010/MFC应用程序框架
VS2010/MFC编程入门之二(利用MFC向导生成单文档应用程序框架)
VS2010/MFC编程入门之三(VS2010应用程序工程中文件的组成结构)
VS2010/MFC编程入门之四(MFC应用程序框架分析)
VS2010/MFC编程入门之五(MFC消息映射机制概述)
第三部分:对话框
VS2010/MFC编程入门之六(对话框:创建对话框模板和修改对话框属性)
VS2010/MFC编程入门之七(对话框:为对话框添加控件)
VS2010/MFC编程入门之八(对话框:创建对话框类和添加控件变量)
VS2010/MFC编程入门之九(对话框:为控件添加消息处理函数)
VS2010/MFC编程入门之十(对话框:设置对话框控件的Tab顺序)
VS2010/MFC编程入门之十一(对话框:模态对话框及其弹出过程)
VS2010/MFC编程入门之十二(对话框:非模态对话框的创建及显示)
VS2010/MFC编程入门之十三(对话框:属性页对话框及相关类的介绍)
VS2010/MFC编程入门之十四(对话框:向导对话框的创建及显示)
VS2010/MFC编程入门之十五(对话框:一般属性页对话框的创建及显示)
VS2010/MFC编程入门之十六(对话框:消息对话框)
VS2010/MFC编程入门之十七(对话框:文件对话框)
VS2010/MFC编程入门之十八(对话框:字体对话框)
VS2010/MFC编程入门之十九(对话框:颜色对话框)
第四部分:常用控件
VS2010/MFC编程入门之二十(常用控件:静态文本框)
VS2010/MFC编程入门之二十一(常用控件:编辑框Edit Control)
VS2010/MFC编程入门之二十二(常用控件:按钮控件Button、Radio Button和Check Box)
VS2010/MFC编程入门之二十三(常用控件:按钮控件的编程实例)
VS2010/MFC编程入门之二十四(常用控件:列表框控件ListBox)
VS2010/MFC编程入门之二十五(常用控件:组合框控件Combo Box)
VS2010/MFC编程入门之二十六(常用控件:滚动条控件Scroll Bar)
VS2010/MFC编程入门之二十七(常用控件:图片控件Picture Control)
VS2010/MFC编程入门之二十八(常用控件:列表视图控件List Control 上)
VS2010/MFC编程入门之二十九(常用控件:列表视图控件List Control 下)
VS2010/MFC编程入门之三十(常用控件:树形控件Tree Control 上)
VS2010/MFC编程入门之三十一(常用控件:树形控件Tree Control 下)
VS2010/MFC编程入门之三十二(常用控件:标签控件Tab Control 上)
VS2010/MFC编程入门之三十三(常用控件:标签控件Tab Control 下)
第五部分:菜单、工具栏与状态栏
VS2010/MFC编程入门之三十四(菜单:VS2010菜单资源详解)
VS2010/MFC编程入门之三十五(菜单:菜单及CMenu类的使用)
VS2010/MFC编程入门之三十六(工具栏:工具栏资源及CToolBar类)
VS2010/MFC编程入门之三十七(工具栏:工具栏的创建、停靠与使用)
VS2010/MFC编程入门之三十八(状态栏的使用详解)
第六部分:文档、视图和框架
VS2010/MFC编程入门之三十九(文档、视图和框架:概述)
VS2010/MFC编程入门之四十(文档、视图和框架:各对象之间的关系)
VS2010/MFC编程入门之四十一(文档、视图和框架:分割窗口)
第七部分:MFC常用类
VS2010/MFC编程入门之四十二(MFC常用类:CString类)
VS2010/MFC编程入门之四十三(MFC常用类:CTime类和CTimeSpan类)
VS2010/MFC编程入门之四十四(MFC常用类:定时器Timer)
VS2010/MFC编程入门之四十五(MFC常用类:CFile文件操作类)
VS2010/MFC编程入门之四十六(MFC常用类:MFC异常处理)
第八部分:字体和文本输出
VS2010/MFC编程入门之四十七(字体和文本输出:CFont字体类)
VS2010/MFC编程入门之四十八(字体和文本输出:文本输出)
第九部分:图形图像
VS2010/MFC编程入门之四十九(图形图像:CDC类及其屏幕绘图函数)
VS2010/MFC编程入门之五十(图形图像:GDI对象之画笔CPen)
VS2010/MFC编程入门之五十一(图形图像:GDI对象之画刷CBrush)
第十部分:Ribbon界面开发
VS2010/MFC编程入门之五十二(Ribbon界面开发:创建Ribbon样式的应用程序框架)
VS2010/MFC编程入门之五十三(Ribbon界面开发:为Ribbon Bar添加控件)
VS2010/MFC编程入门之五十四(Ribbon界面开发:使用更多控件并为控件添加消息处理函数)
在 解决方案视图 中看到,此工程的文件要比单文档应用程序少的多,在 Class View 中主要有三个类:CAboutDlg、CAdditionApp 和 CAdditionDlg 。
打开资源视图
在Resource View视图中可以看到工程 Addition 的资源树,展开Addition.rc,下面有四个子项:Dialog(对话框)、Icon(图标)、String Table(字符串表)和Version(版本)。然后展开Dialog项,下面有两个对话框模板,其ID分别为:IDD_ABOUTBOX和IDD_ADDITION_DIALOG,前者是“关于”对话框的模板,后者是主对话框的模板。ID是资源的唯一标识,本质上是一个无符号整数,一般ID代表的整数值由系统定义,我们无需干涉。
可见对于主对话框来说,创建对话框第一步中的创建新的对话框模板已经由系统自动完成了。如果需要再添加创建新的对话框模板时,需要在Resource View 的“Dialog”节点上点右键,在右键菜单中选择“Insert Dialog”,就会生成新的对话框模板,并且会自动分配ID。如图:
在Addition对话框模板上点右键,然后在右键菜单中选择Properties,则在右侧面板中会显示对话框的属性列表。如下图:
说明:
修改了标题属性。这时我们运行此程序后的界面如下:
对于自动添加的一个标题为“TODO:Place dialog controls here.”的静态文本框,这里将它删掉,继续添加新的静态文本框。
删除控件时,可以使用鼠标左键点击选中它,选中后控件的周围会出现虚线框,然后按Delete键就可以将其删除了。在“Addition”工程的Resource View中打开上一讲中创建的对话框模板IDD_ADDITION_DIALOG,自动添加的静态文本框就可以使用这种方法删除。
在添加新的静态文本框以前,先看看Toolbox视图是否显示了,如果没有显示,在菜单栏上点击 View->Toolbox 即可。Toolbox视图如下图:
Toolbox(工具箱) 中列出了一些常用控件,其中有一个是 Static Text,即是我们要添加的控件。在 Toolbox 中的 Static Text 上点下鼠标左键不放开,并拖到 IDD_ADDITION_DIALOG 对话框模板上,模板上会出现一个虚线框,我们找到合适的位置松开鼠标左键放下它。
用鼠标左键选中控件后周围出现虚线框,然后鼠标移到虚线框上几个黑点的位置会变成双向箭头的形状,此时就可以按下鼠标左键并拖动来改变控件大小了。我们可以这样改变新添加的静态文本框控件的大小,以更好的显示标题。当然,整个对话框模板也可以用这种方法改变大小。
接下来就该修改静态文本框的文字了。鼠标右键点击静态文本框,在右键菜单中选择 “Properties”,Properties 面板就会显示出来,在面板上修改 Caption 属性为 “被加数”,ID 修改为 IDC_SUMMAND_STATIC。此时模板如下图:
添加编辑框的过程与静态文本框类似,在Toolbox中选中Edit Control控件拖到对话框模板上,并使其与之前的静态文本框水平对齐(为了美观),然后调整其大小使之适合被加数的输入。
在编辑框上点右键,仍然在右键菜单中选择“Properties”显示出属性(Properties)面板,修改其ID为IDC_SUMMAND_EDIT。此时模板如下图:
3. 按照1的方法添加一个标题为“加数”的静态文本框,用于显示字符串--“加数”。并将其ID改为IDC_ADDEND_STATIC。
4. 按照2的方法添加一个ID为IDC_ADDEND_EDIT的编辑框,用来输入加数。
5. 按照1的方法添加一个标题为“和”的静态文本框,用于显示文字--“和”。并修改其ID为IDC_SUM_STATIC。
6. 按照2的方法添加一个ID为IDC_SUM_EDIT的编辑框,用来显示最终的加和。
7. 类似的添加按钮(Button)控件到对话框模板,用于在被点击后触发加法计算。修改其标题为“计算”,ID为IDC_ADD_BUTTON。
到此,对话框模板如图:
8.删除 确定 按钮。打开 取消 按钮的属性面板,将标题改为 “退出”,并使其与“计算”按钮水平对齐。
9.根据控件的布局,适当调整整个对话框模板的大小,使其相对控件布局来说大小合适,界面美观。
这样在对话框模板中就把我们在本例中需要用到的控件就添加完了。最终效果如下:
运行截图:
因为 程序Addition 是基于对话框的程序,所以程序自动创建了对话框模板 IDD_ADDITION_DIALOG,并自动生成了对话框类CAdditionDlg,它是从 CDialogEx 类派生的。大家用过VC++ 6.0的可能记得,我们定义的对话框类都是从CDialog类派生的,但在VS2010中,一般对话框类都是继承自CDialogEx类。
如果是自己新添加的对话框模板,怎样为它创建对话框类呢?
3.选择“Add Class”后会弹出一个对话框,在对话框中“Class name”下的编辑框中写入自定义的类名就可以了,例如CMyDialog。
4.最后点“Finish”完成。
最终你就可以在Class View中看到新生成的对话框类CMyDialog了,并且在Solution Explorer中有相应的MyDialog.h头文件和MyDialog.cpp源文件生成。CMyDialog类同样派生于CDialogEx类。
注意,一般类名都以C打头,又比如,CTestDlg。
上面步骤中 为对话框添加了几个控件,包括三个静态文本框,三个编辑框,一个按钮控件。程序自动生成的 取消 按钮保留,作为退出按钮,而 确定 按钮删除掉了。
静态文本框只是为了说明后面紧跟的编辑框中数据的意义,是被加数、加数还是和,所以它们是不会变的,我们就不为它们添加变量了。按钮控件是用来操作的,这里也不为它们添加变量。编辑框中的数据可能会经常变化,有必要为它们每个控件关联一个变量。
首先为被加数的编辑框 IDC_SUMMAND_EDIT 添加变量。
5.点“Finish”完成。
注意,类的成员变量名一般以m_打头,以标识它是一个成员变量。
参照此方法,再分别为加数的编辑框IDD_ADDEND_EDIT添加double型变量m_editAddend、和的编辑框IDD_SUM_EDIT添加double型变量m_editSum。
在程序运行界面中,用户往往会改变控件的属性,例如,在编辑框中输入字符串,或者改变组合框的选中项,又或者改变复选框的选中状态等。控件的属性改变后MFC会相应修改控件关联变量的值。这种同步的改变是通过MFC为对话框类自动生成的成员函数DoDataExchange()来实现的,这也叫做对话框的数据交换和检验机制。
我们为三个编辑框添加了变量以后,在AdditionDlg.cpp 中 CAdditionDlg 的 DoDataExchange()函数的函数体中多了三条DDX_Text调用语句。下面是函数体代码和鸡啄米添加的注释。
再以 Addition 程序为例简单说下数据交换机制。如果我们在程序运行界面中输入被加数,则通过CAddition的DoDataExchange()函数可以将输入的值保存到 m_editSummand 变量中,反之如果程序运行中修改了变量 m_editSummand 的值,则通过 CAddition 的 DoDataExchange() 函数也可以将新的变量值显示到被加数的编辑框中。
但是这种数据交换机制中,DoDataExchange() 并不是被自动调用的,而是需要我们在程序中调用 CDialogEx::UpdateData()函数,由 UpdateData() 函数再去自动调用 DoDataExchange()的。
CDialogEx::UpdateData() 函数的原型为:BOOL UpdateData(BOOL bSaveAndValidate = TRUE);
参数:bSaveAndValidate 用于指示数据传输的方向,TRUE 表示从 控件 传给 变量,FALSE 表示从 变量 传给 控件。默认值是 TRUE,即从 控件 传给 变量。
返回值:CDialogEx::UpdateData()函数的返回值表示操作是否成功,成功则返回TRUE,否则返回FALSE。
在下一讲中将具体演示CDialogEx::UpdateData()函数如何使用。
鸡啄米本节主要讲的是新建对话框类和添加控件变量,控件的消息处理函数将在下一讲详细介绍。
创建对话框类和添加控件变量在上面已经讲过,这一讲的主要内容是如何为控件添加消息处理函数。
MFC为对话框和控件等定义了诸多消息,我们对它们操作时会触发消息,这些消息最终由消息处理函数处理。比如我们点击按钮时就会产生 BN_CLICKED 消息,修改编辑框内容时会产生 EN_CHANGE消息等。一般为了让某种操作达到效果,我们只需要实现某个消息的消息处理函数。
鸡啄米仍以前面的加法计算器的程序为例,说明怎样为“计算”按钮控件添加消息处理函数。添加方法列出4种:
用过的VC++ 6.0的朋友应该对Class Wizard很熟悉了,添加类、消息处理函数等经常会用到它,可以说是一个很核心的功能。但从VS2002开始就见不到Class Wizard了,大部分功能都集成到对话框和控件等的属性中了,使用很方便。到VS2010,久违的Class Wizard又回来了。但鸡啄米已经习惯了使用属性中的功能了,对于从VC++ 6.0直接转VS2010的朋友可能觉得还是使用Class Wizard比较习惯。
大家应该记得,“计算” 按钮的 ID 为 IDC_ADD_BUTTON,上图中 命令 标签下,对象 ID 列表中有此 ID,因为我们是想实现点击按钮后的消息处理函数,所以在 消息 列表中选择 BN_CLICKED 消息,然后点右上方的 添加处理程序 就可以添加BN_CLICKED 消息处理函数 OnClickedAddButton 了。当然你也可以改名,但一般用的默认的就可以。
在 “计算” 按钮上点右键,然后在右键菜单中选择菜单项“Add Event Handler...”,弹出 “Event Handler Wizard” 对话框,如下图:
可见 “消息类型” 中默认选中的就是 BN_CLICKED 消息,函数名 和 所在 类 都已经自动给出,直接点 “确定” 就可以了。
上面说过,从VS2002开始就主要从 属性视图 添加 消息处理函数了。我们在 “计算” 按钮上点右键,在右键菜单中选择“Properties”,右侧面板中会显示按钮的属性视图。
我们可以像上图中那样,点属性视图的 “控制事件” 按钮(类似 闪电标志),下面列出了 “计算” 按钮的所有消息。我们要处理的是 BN_CLICKED 消息,点其右侧空白列表项,会出现一个带下箭头的按钮,再点此按钮会出现 “
最直接最简单的方法就是,双击 “计算” 按钮,MFC会自动为其在CAdditionDlg类中添加 BN_CLICKED 消息的处理函数OnBnClickedAddButton()。
在我们使用任意一种方法添加了消息处理函数以后,都只能得到一个空的OnBnClickedAddButton()函数的函数体,要实现我们想要的功能,还需要在函数体中加入自定义功能代码。
在加法计算器程序中,我们想要“计算”按钮实现的功能是,获取被加数和加数的数值,然后计算它们的和并显示到和的编辑框里。那么,OnBnClickedAddButton()的函数体就应修改为:
-
void CAdditionDlg::OnClickedAddButton()
-
{
-
// TODO: 在此添加控件通知处理程序代码
-
-
// 将各控件中的数据保存到相应的变量
-
UpdateData(TRUE);
-
-
// 将被加数和加数的加和赋值给m_editSum
-
m_editSum = m_editSummand + m_editAddend;
-
-
// 根据各变量的值更新相应的控件。和的编辑框会显示m_editSum的值
-
UpdateData(FALSE);
-
}
上面的代码中已经添加注释,大家应该很容易理解了。对于UpdateData()函数的说明在上一讲中已经介绍过,如果忘了可以再回上一讲了解了解。接下来我们运行下此应用程序。在运行结果界面中,输入被加数 1.1,加数 2.2 ,然后点“计算”:
简单分析下运行过程:输入被加数和加数,点“计算”按钮后产生点击消息,从而调用 OnBnClickedAddButton()函数。进入此函数后,首先由 UpdateData(TRUE)函数将被加数的值5.1和加数的值2.3分别保存到变量m_editSummand和m_editAddend,然后通过语句m_editSum = m_editSummand + m_editAddend;计算出被加数和加数的和为7.4,并把7.4赋值给m_editSum。最后调用UpdateData(FALSE) 根据被加数、加数、和的值更新三个编辑框的显示值,就得到了上图中的结果。
到此,一个具有简单的加法运算功能的加法计算器应用程序就基本完成了。
上一讲为“计算”按钮添加了消息处理函数后,加法计算器已经能够进行浮点数的加法运算。但是还有个遗留的小问题,就是对话框控件的Tab顺序问题。
运行加法计算器程序,显示对话框后不进行任何操作,直接按回车,可以看到对话框退出了。这是因为“退出”按钮是Tab顺序为1的控件,也就是第一个接受用户输入的控件。但是按照我们的输入习惯,应该是被加数的编辑框首先接受用户输入,然后是加数编辑框,再接下来是“计算”按钮,最后才是“退出”按钮。
先直观的看看各个控件的Tab顺序:打开 “资源视图” ,然后在资源中找到对话框 IDD_ADDITION_DIALOG,双击ID后中间客户区域出现其模板视图。在主菜单中选择 “Format”->"Tab Order",或者按快捷键Ctrl+D,对话框模板上就会显示各个控件的Tab顺序数字。如下图:
上图中每个控件左上角都有一个数字,这就是它的 Tab 响应顺序。对话框刚打开时输入焦点就在Tab顺序为1的“退出”按钮上,不做任何操作按下Tab键,输入焦点就会转移到Tab顺序为2的“被加数”静态文本框上,但是因为静态文本框不接受任何输入,所以输入焦点继续自动转移到Tab顺序为3的被加数编辑框,再按Tab键,输入焦点又会转移到Tab顺序为4的“加数”静态文本框上,同样由于它是静态文本框,输入焦点不停留继续转移到加数编辑框,后面的控件同理。
我们认为这个顺序不合理,那怎么修改呢?很简单,从自己认为Tab顺序应该为1的控件开始依次单击,随着单击的完成,各控件的Tab响应顺序也按我们的想法设置好了。
例如,此例中我们可以依次单击被加数编辑框、“被加数”静态文本框、加数编辑框、“加数”静态文本框、和编辑框、“和”静态文本框、“计算”按钮和“退出”按钮。设置完后如下图:
最后按ESC键,确认设置并退出对话框模板的Tab顺序设置状态。
现在我们再运行程序,可以看到对话框打开后最初的输入焦点在被加数编辑框上,然后我们按Tab键,输入焦点移到加数编辑框上,继续多次按Tab键时,输入焦点会按“和编辑框--‘计算’按钮--‘退出’按钮--被加数编辑框--加数编辑框--和编辑框......”的顺序循环转移。这样就达到了我们的目的。
一 . 模态对话框 和 非模态对话框
Windows对话框分为两类:模态对话框 和 非模态对话框。
大家对Windows系统很了解,相信这两种对话框应该都遇到过。之前的加法计算器对话框其实就是模态对话框。
二 . 模态对话框是怎样弹出的
毕竟加法计算器程序大部分都是MFC自动生成的,对话框怎么弹出来的大家可能还不是很清楚。鸡啄米下面简单说说它是在哪里弹出来的,再重新建一个新的对话框并弹出它,这样大家实践以后就能更灵活的使用模态对话框了。
大家打开 Addition.cpp 文件,可以看到 CAdditionApp 类有个 InitInstance() 函数,在 MFC应用程序框架分析 中提到过此函数,不过那是单文档应用程序 App类 中的,函数体不太相同,但都是进行App类实例的初始化工作。
InitInstance() 函数的 后半部分有一段代码就是定义对话框对象并弹出对话框的,鸡啄米下面给出这段代码并加以注释:
-
CAdditionDlg dlg;
// 定义对话框类CAdditionDlg的对象dlg
-
m_pMainWnd = &dlg;
// 将dlg设为主窗口
-
INT_PTR nResponse = dlg.
DoModal();
// 弹出对话框dlg,并将DoModal函数的返回值(退出时点击按钮的ID)赋值给nResponse
-
if (nResponse == IDOK)
// 判断返回值是否为OK按钮(其ID为IDOK,鸡啄米已经将它删除)
-
{
-
// TODO: 在此放置处理何时用
-
// “确定”来关闭对话框的代码
-
}
-
else
if (nResponse == IDCANCEL)
// 判断返回值是否为Cancel按钮(其ID为IDCANCEL,鸡啄米将它的Caption改为了“退出”)
-
{
-
// TODO: 在此放置处理何时用
-
// “取消”来关闭对话框的代码
-
}
-
else
if (nResponse ==
-1)
-
{
-
TRACE(traceAppMsg,
0,
"警告: 对话框创建失败,应用程序将意外终止。\n");
-
TRACE(traceAppMsg,
0,
"警告: 如果您在对话框上使用 MFC 控件,则无法 #define _AFX_NO_MFC_CONTROLS_IN_DIALOGS。\n");
-
}
弹出对话框比较关键的一个函数,就是对话框类的 DoModal() 函数。CDialog::DoModal() 函数的原型为:
virtual INT_PTR DoModal();
返回值:整数值,指定了传递给CDialog::EndDialog(该函数用于关闭对话框)的nResult参数值。如果函数不能创建对话框,则返回-1;如果出现其它错误,则返回IDABORT。
调用了它,对话框就会弹出,返回值是退出对话框时所点的按钮的 ID,比如,我们点了“退出”按钮,那么 DoModal 返回值为IDCANCEL。
三 . 添加一个新对话框并弹出它
为加法计算器程序添加一个对话框,在计算之前询问用户是否要进行计算。大家可以完整的看下对话框的添加和弹出过程。
1.在 Resource View 中的 “Dialog” 上点右键选择 “Insert Dialog”,创建一个新的对话框模板,修改其ID为IDD_TIP_DIALOG,Caption改为“提示”,然后参考 “为对话框添加控件” 中所讲,在对话框模板上添加一个静态文本框(static text),Caption改为“您确定要进行加法计算吗?”,接下来修改OK按钮的Caption为“确定”,Cancel按钮的Caption为“取消”,最后调整各个控件的位置和对话框的大小。最终的对话框模板如下图:
2.根据“创建对话框类和添加控件变量”中创建对话框类的方法,在对话框模板上点右键选择“Add Class...”,弹出添加类的对话框,设置“Class name”为CTipDlg,点“OK”。在Solution Explorer中可以看到生成了CTipDlg类的头文件TipDlg.h和源文件TipDlg.cpp。
3.我们要在点“计算”按钮之后弹出此提示对话框,那么就要在“计算”按钮的消息处理函数OnBnClickedAddButton()中访问提示对话框类,所以为了访问CTipDlg类,在AdditionDlg.cpp中包含CTipDlg的头文件:#include "CTipDlg.h"。
4.修改OnBnClickedAddButton()的函数体,在所有代码前,构造CTipDlg类的对象tipDlg,并通过语句tipDlg.DoModal();弹出对话框,最后判断DoModal()函数的返回值是IDOK还是IDCANCEL来确定是否继续进行计算。OnBnClickedAddButton()函数修改后如下:
-
void CAdditionDlg::OnClickedAddButton()
-
{
-
// TODO: Add your control notification handler code here
-
INT_PTR nRes;
// 用于保存DoModal函数的返回值
-
-
CTipDlg tipDlg;
// 构造对话框类CTipDlg的实例
-
nRes = tipDlg.
DoModal();
// 弹出对话框
-
if (IDCANCEL == nRes)
// 判断对话框退出后返回值是否为IDCANCEL,如果是则return,否则继续向下执行
-
return;
-
-
// 将各控件中的数据保存到相应的变量
-
UpdateData(TRUE);
-
-
// 将被加数和加数的加和赋值给m_editSum
-
m_editSum = m_editSummand + m_editAddend;
-
-
// 根据各变量的值更新相应的控件。和的编辑框会显示m_editSum的值
-
UpdateData(FALSE);
-
}
5.测试。编译运行程序后,在对话框上输入被加数和加数,点“计算”,弹出提示对话框询问是否进行计算,如果选择“确定”,则提示对话框退出,并在主对话框上显示被加数和加数的和,而如果选择“取消”,则提示对话框也会退出,但主对话框显示的和不变,即没有进行加法计算。
总结:主要步骤是,创建新对话框后,关联一个类,然后用一个指针指向该类;接着用该指针创建对话框和显示对话框,最后在虚构函数中销毁该指针,释放资源。
非模态对话框显示后,程序其他窗口仍能正常运行,可以响应用户输入,还可以相互切换。将上一讲中创建的Tip模态对话框改为非模态对话框,让大家看下效果。
实际上,模态对话框 和 非模态对话框 在创建对话框资源和生成对话框类上是没有区别的,所以上一讲中创建的IDD_TIP_DIALOG对话框资源和 CTipDlg 类都不需要修改。
需要修改的是,对话框类实例的创建和显示,也就是之前在 CAdditionDlg::OnBnClickedAddButton() 函数体中添加的对话框显示代码。下面是具体步骤:
-
void CAdditionDlg::OnClickedAddButton()
-
{
-
// TODO: Add your control notification handler code here
-
/*INT_PTR nRes; // 用于保存DoModal函数的返回值
-
-
CTipDlg tipDlg; // 构造对话框类CTipDlg的实例
-
nRes = tipDlg.DoModal(); // 弹出对话框
-
if (IDCANCEL == nRes) // 判断对话框退出后返回值是否为IDCANCEL,如果是则return,否则继续向下执行
-
return;*/
-
-
// 如果指针变量m_pTipDlg的值为NULL,则对话框还未创建,需要动态创建
-
if (
NULL == m_pTipDlg)
-
{
-
// 创建非模态对话框实例
-
m_pTipDlg =
new
CTipDlg();
-
m_pTipDlg->
Create(IDD_TIP_DIALOG,
this);
-
}
-
// 显示非模态对话框
-
m_pTipDlg->
ShowWindow(SW_SHOW);
-
-
// 将各控件中的数据保存到相应的变量
-
UpdateData(TRUE);
-
-
// 将被加数和加数的加和赋值给m_editSum
-
m_editSum = m_editSummand + m_editAddend;
-
-
// 根据各变量的值更新相应的控件。和的编辑框会显示m_editSum的值
-
UpdateData(FALSE);
-
}
这样,非模态对话框创建和显示的代码就添加修改完了。让我们运行下看看效果吧。
在加法计算器对话框上输入被加数和加数,然后点“计算”按钮,依然像上节一样弹出了提示对话框,但是先不要关闭它,你可以拖动它后面的加法计算器对话框试试,我们发现加法计算器对话框竟然可以拖动了,而且“和”编辑框里已经显示了运算结果,这表明提示对话框显示以后还没有关闭,OnBnClickedAddButton() 就继续向下执行了,不仅如此,加法计算器的每个编辑框还都可以响应输入。
这只是一个简单的例子,非模态对话框的用处有很多,以后大家在软件开发中会用到。
属性页对话框的分类
属性页对话框想必大家并不陌生,XP系统中桌面右键点属性,弹出的就是属性页对话框,它通过标签切换各个页面。另外,我们在创建MFC工程时使用的向导对话框也属于属性页对话框,它通过点击“Next”等按钮来切换页面。
属性页对话框就是包含一般属性页对话框和向导对话框两类。它将多个对话框集成于一身,通过标签或按钮来切换页面。
属性页对话框相关类
我们使用属性页对话框时,用到的类主要有两个:CPropertyPage类 和 CPropertySheet 类。
1.CPropertyPage 类
CPropertyPage类继承自CDialog类,它被用于处理某单个的属性页,所以要为每个属性页都创建一个继承自CPropertyPage的子类。大家可以在VS2010的MSDN中查找CPropertyPage类以及它的成员的详细说明。下面鸡啄米就为大家讲解MSDN中列出的CPropertyPage类的部分主要成员函数。
-
(
1)构造函数
-
这里讲三个CProperty类的构造函数,函数原型为:
-
CPropertyPage( );
-
-
explicit CPropertyPage(
-
UINT nIDTemplate,
-
UINT nIDCaption =
0,
-
DWORD dwSize =
sizeof(PROPSHEETPAGE)
-
);
-
-
explicit CPropertyPage(
-
LPCTSTR lpszTemplateName,
-
UINT nIDCaption =
0,
-
DWORD dwSize =
sizeof(PROPSHEETPAGE)
-
);
-
-
第一个是没有任何参数的构造函数。
-
第二个构造函数中,参数nIDTemplate是属性页的对话框资源ID,
-
参数nIDCaption是属性页对话框选项卡的标题所用字符串资源的ID,
-
若设为
0,则选项卡标题就使用该属性页的对话框资源的标题。
-
-
第三个构造函数中,参数lpszTemplateName为属性页的对话框资源的名称字符串,
-
不能为
NULL。参数nIDCaption同上。
-
-
(
2)CancelToClose()函数
-
在模态属性页对话框的属性页进行了某不可恢复的操作后,使用CancelToClose() 函数
-
将“OK”按钮改为“Close”按钮,并禁用“Cancel”按钮。函数原型为:
-
void CancelToClose( );
-
-
(
3)SetModified()函数
-
调用此函数可激活或禁用“Apply”按钮,函数原型为:
-
void SetModified(
BOOL bChanged =
TRUE);
-
-
(
4)可重载函数
-
CPropertyPage类提供了一些消息处理函数,来响应属性页对话框的各种消息。
-
我们重载这些消息处理函数,就可以自定义对属性页对话框操作的处理。
-
可重载的消息处理函数包括:
-
OnApply:处理属性页的“Apply”按钮被单击的消息
-
OnCancel:处理属性页的“Cancel”按钮被单击的消息
-
OnKillActive:处理属性页当前活动状态被切换的消息,常用于数据验证
-
OnOK:处理属性页的“OK”按钮、“Apply”按钮或者“Close”按钮被单击的消息
-
OnQueryCancel:处理属性页的“Cancel”按钮被单击前发出的消息
-
OnReset:处理属性页的“Reset”按钮被单击的消息
-
OnSetActive:处理属性页被切换为当前活动页的消息
-
OnWizardBack:处理属性页的“Back”按钮被单击的消息,仅在向导对话框中有效
-
OnWizardFinish:处理属性页的“Finish”按钮被单击的消息,仅在向导对话框中有效
-
OnWizardNext:处理属性页的“Next”按钮被单击的消息,仅在向导对话框中有效
2.CPropertySheet类
CPropertySheet类继承自CWnd类,它是属性表类,负责加载、打开或删除属性页,并可以在属性页对话框中切换属性页。它跟对话框类似,也有模态和非模态两种。下面鸡啄米就讲解CPropertySheet类的部分成员函数。
-
(
1)构造函数
-
这里依然列出CPropertySheet类的三个构造函数:
-
CPropertySheet( );
-
-
explicit CPropertySheet(
-
UINT nIDCaption,
-
CWnd* pParentWnd = NULL,
-
UINT iSelectPage = 0
-
);
-
-
explicit CPropertySheet(
-
LPCTSTR pszCaption,
-
CWnd* pParentWnd = NULL,
-
UINT iSelectPage = 0
-
);
-
-
参数 nIDCaption:标题的字符串资源的ID。
-
参数 pParentWnd:属性页对话框的父窗口,若设为
NULL,则父窗口为应用程序的主窗口。
-
参数 iSelectPage:初始状态时,活动属性页的索引,默认为第一个添加到属性表的属性页。
-
参数 pszCaption:标题字符串。
-
-
(
2)
GetActiveIndex()函数
-
获取当前活动属性页的索引。函数原型为:
-
int GetActiveIndex( ) const;
-
返回值:当前活动属性页的索引。
-
-
(
3)
GetActivePage()函数
-
获取当前活动属性页对象。函数原型为:
-
CPropertyPage* GetActivePage( ) const;
-
返回值:当前活动属性页对象的指针。
-
-
(
4)
GetPage()函数
-
获取某个属性页对象。函数原型为:
-
CPropertyPage* GetPage(int nPage) const;
-
参数nPage:目标属性页的索引。
-
返回值:目标属性页对象的指针。
-
-
(
5)
GetPageCount()函数
-
获取属性页的数量。函数原型为:
-
int GetPageCount( ) const;
-
返回值:属性页的数量。
-
-
(
6)
GetPageIndex()函数
-
获取某属性页在属性页对话框中的索引。函数原型为:
-
int GetPageIndex(CPropertyPage* pPage);
-
参数pPage:要获取索引的属性页对象的指针。
-
返回值:属性页对象在属性页对话框中的索引。
-
-
(
7)
SetActivePage()函数
-
设置某个属性页为活动属性页。函数原型为:
-
BOOL SetActivePage(
-
int nPage
-
);
-
-
BOOL SetActivePage(
-
CPropertyPage* pPage
-
);
-
参数 nPage:要设置为活动属性页的索引。
-
参数 pPage:要设置为活动属性页的对象指针。
-
-
(
8)
SetWizardButtons()函数
-
在向导对话框上启用或禁用Back、Next或Finish按钮,应在调用DoModal之前调用此函数。
-
函数原型为:
-
void SetWizardButtons(
-
DWORD dwFlags
-
);
-
-
参数dwFlags:设置向导按钮的外观和功能属性。可以是以下值的组合:
-
PSWIZB_BACK 启用“Back”按钮,如果不包含此值则禁用“Back”按钮。
-
PSWIZB_NEXT 启用“Next”按钮,如果不包含此值则禁用“Next”按钮。
-
PSWIZB_FINISH 启用“Finish”按钮。
-
PSWIZB_DISABLEDFINISH 显示禁用的“Finish”按钮。
-
-
(
9)
SetWizardMode()函数
-
设置属性页对话框为向导对话框模式,应在调用DoModal之前调用此函数。函数原型为:
-
void SetWizardMode( );
-
-
(
10)
SetTitle()函数
-
设置属性对话框的标题。函数原型为:
-
void SetTitle(
-
LPCTSTR lpszText,
-
UINT nStyle = 0
-
);
-
参数lpszText:标题字符串。
-
参数nStyle:指定属性表标题的风格。应当为
0 或 PSH_PROPTITLE。
-
如果设为PSH_PROPTITLE,则单词“Properties”会出现在指定标题之后。
-
例如,
SetTitle(
"Simple",PSH_PROPTITLE)这种调用会使得属性表标题为 “Simple Properties”。
-
-
(
11)
AddPage()函数
-
为属性对话框添加新的属性页。函数原型为:
-
void AddPage(
-
CPropertyPage *pPage
-
);
-
参数pPage:要添加的新的属性页的对象指针。
-
-
(
12)
PressButton()函数
-
模拟按下某指定的按钮。函数原型为:
-
void PressButton(
-
int nButton
-
);
-
参数nButton:要模拟按下的按钮,它可以是下列值之一:
-
PSBTN_BACK 选择“Back”按钮。
-
PSBTN_NEXT 选择“Next”按钮。
-
PSBTN_FINISH 选择“Finish”按钮。
-
PSBTN_OK 选择“OK”按钮。
-
PSBTN_APPLYNOW 选择“Apply”按钮。
-
PSBTN_CANCEL 选择“Cancel”按钮。
-
PSBTN_HELP 选择“帮助”按钮。
-
-
(
13)
RemovePage()函数
-
删除某属性页。函数原型为:
-
void RemovePage(
-
CPropertyPage *pPage
-
);
-
-
void RemovePage(
-
int nPage
-
);
-
-
参数pPage:要删除的属性页的对象指针。
-
参数nPage:要删除的属性页的索引。
属性对话框和相关的两个类鸡啄米就先介绍到这,主要是为后面使用属性页对话框做准备。
仍然以前面的“加法计算器”的例子为基础,在其中加入向导对话框,我们可以用它来说明加法计算器的使用方法,一步一步引导用户操作,这也是比较常见的用法。
加法计算器使用时大概可以分为三步:输入被加数、输入加数、点“计算”按钮。
鸡啄米就详细说明向导对话框的创建步骤:
1. 创建属性页对话框资源
根据创建对话框模板和修改对话框属性中所讲方法,在“Resource View”的Dialog”节点上点右键,然后在右键菜单中选择“Insert Dialog”创建第一个对话框模板,对话框的ID属性设置为IDD_SUMMAND_PAGE,Caption属性改为“被加数页”,Style属性在下拉列表中选择“Child”,Border属性在下拉列表中选择“Thin”。
删除“OK”和“Cancel”按钮,再按照为对话框添加控件中所讲方法,添加一个静态文本框,并修改静态文本框的Caption属性为“请先输入double型被加数”。
按照上述步骤,继续添加第二个和第三个对话框资源。第二个对话框模板的ID设为IDD_ADDEND_PAGE,Caption属性改为“加数页”,也添加一个静态文本框,Caption设为“请继续输入double型加数”,其他属性同第一个对话框。第三个对话框模板的ID设为IDD_ADD_PAGE,Caption属性改为“计算页”,添加静态文本框的Caption属性改为“最后请按下“计算”按钮”,其他属性也第一个对话框一样。
2. 创建属性页类
按照创建对话框类和添加控件变量中的方法,在第一个对话框模板上点右键,在右键菜单中选择“Add Class”,弹出类向导对话框,在“Class name”编辑框中输入类名“CSummandPage”,与之前不同的是,因为属性页类都应继承于CPropertyPage类,所以要修改下面“Base class”的选项,在下拉列表中选择“CPropertyPage”。
因为是第一个属性页,所以它应该有一个“下一步”按钮,在哪里添加呢?上一讲CPropertyPage类的可重载函数中提到,OnSetActive函数用于处理属性页被切换为当前活动页的消息,所以我们可以在OnSetActive函数中进行相关设置。
那怎样重载OnSetActive函数呢?我们可以在“Class View”中找到“CSummandPage”节点,点右键弹出右键菜单,选择“Properties”,然后VS2010右侧面板上会显示对话框的属性列表,属性列表的工具栏上有个tip信息为“Overrides”的按钮,按下它,下方列表中就列出了重载函数,找到“OnSetActive”,点其右侧空白列表项出现向下箭头,再点箭头就在下面出现了“
我们只需在OnSetActive函数体中添加相关代码就可以实现添加“下一步”按钮的效果了。新的函数体如下:
-
BOOL CSummandPage::OnSetActive()
-
{
-
// TODO: Add your specialized code here and/or call the base class
-
-
// 获得父窗口,即属性表CPropertySheet类
-
CPropertySheet* psheet = (CPropertySheet*)
GetParent();
-
// 设置属性表只有“下一步”按钮
-
psheet->
SetWizardButtons(PSWIZB_NEXT);
-
-
return CPropertyPage::
OnSetActive();
-
}
为第二个和第三个对话框也分别添加属性页类CAddendPage和CAddPage。但第二个对话框的属性页不需要重载OnSetActive函数。第三个对话框是最后一个对话框,所以不需要“下一步”按钮,而应该换成“完成”按钮,所以也需要重载OnSetActive函数设置“完成”按钮。重载后的OnSetActive如下:
-
BOOL CAddPage::OnSetActive()
-
{
-
// TODO: Add your specialized code here and/or call the base class
-
-
// 获得父窗口,即属性表CPropertySheet类
-
CPropertySheet* psheet = (CPropertySheet*)
GetParent();
-
//设置属性表只有“完成”按钮
-
psheet->
SetFinishText(_T(
"完成"));
-
-
return CPropertyPage::
OnSetActive();
-
}
上面的代码段中,字符串“完成”前加了个_T,这是因为本工程创建的时候用的默认的Unicode字符集,而如果“完成”前不加_T就是ASCII字符串。_T实际上是一个宏,工程的字符集选择为Unicode时字符串就转为Unicode字符串,选择为Muli-Byte时就转为ASCII字符串。我们可以在Solution Explorer的Addition根节点上点右键,在右键菜单上选择“Properties”,弹出工程的属性对话框,Configuration Properties->General右侧列表中的Character Set就显示选择的字符集。
那点了第三个属性页上的“完成”按钮我们想进行某些处理的话,就重载OnWizardFinish函数,方法同OnSetActive函数。重载后的OnWizardFinish函数如下:
-
BOOL CAddPage::OnWizardFinish()
-
{
-
// TODO: Add your specialized code here and/or call the base class
-
-
// 提示向导完成
-
MessageBox(_T(
"使用说明向导已阅读完!"));
-
-
return CPropertyPage::
OnWizardFinish();
-
}
3. 创建属性表类
属性页资源和属性页类创建完以后,还不能生成向导对话框,我们还需要一个属性表类,来容纳这些属性页。
在Solution Explorer视图中的根节点“Addition”上点右键,在右键菜单中选择Add->Class,弹出“Add Class”对话框,然后在中间区域中选择“MFC Class”,点“Add”按钮,弹出另一个类向导对话框,设置Class name为CAddSheet,Base class选择“CPropertySheet”,点“Finish”按钮,这样就属性表类就建好了。
接下来,在新生成的AddSheet.h中包含三个属性页类的头文件:
#include "SummandPage.h"
#include "AddendPage.h"
#include "AddPage.h"
之后在AddSheet.h中添加private变量:
CSummandPage m_summandPage;C
CAddendPage m_addendPage;
CAddPage m_addPage;
然后在AddSheet.cpp文件中修改CAddSheet的两个构造函数为:
-
CAddSheet::
CAddSheet(
UINT nIDCaption, CWnd* pParentWnd,
UINT iSelectPage)
-
:CPropertySheet(nIDCaption, pParentWnd, iSelectPage)
-
{
-
// 添加三个属性页到属性表
-
AddPage(&m_summandPage);
-
AddPage(&m_addendPage);
-
AddPage(&m_addPage);
-
}
-
-
CAddSheet::
CAddSheet(LPCTSTR pszCaption, CWnd* pParentWnd,
UINT iSelectPage)
-
:CPropertySheet(pszCaption, pParentWnd, iSelectPage)
-
{
-
// 添加三个属性页到属性表
-
AddPage(&m_summandPage);
-
AddPage(&m_addendPage);
-
AddPage(&m_addPage);
-
}
4. 显示向导对话框
我们在加法计算器对话框上添加一个按钮,点击它就打开向导对话框。此按钮的ID设为IDC_INSTRUCT_BUTTON,Caption属性设为“使用说明”。
按照为控件添加消息处理函数中所讲方法,为IDC_INSTRUCT_BUTTON按钮在CAdditionDlg类中添加点击消息的处理函数OnBnClickedInstructButton。然后在AdditionDlg.cpp文件中包含CAddSheet的头文件:#include "AddSheet.h"。最后修改OnBnClickedInstructButton函数如下:
-
void CAdditionDlg::OnBnClickedInstructButton()
-
{
-
// TODO: Add your control notification handler code here
-
-
// 创建属性表对象
-
CAddSheet sheet(_T(""));
-
// 设置属性对话框为向导对话框
-
sheet.
SetWizardMode();
-
// 打开模态向导对话框
-
sheet.
DoModal();
-
}
到此,向导对话框就完整的创建完成了,并可以在加法计算器对话框上点 “使用说明” 按钮显示出来。我们来看看效果吧:
上图只是被加数页的效果,点其上“下一步”按钮就可以继续显示后面的两个页面。
是不是向导对话框没有以前想象的那般复杂了?大家可以发挥想象,进行更复杂的修改,实现更完善的功能。依然欢迎朋友们到鸡啄米博客来交流学习。
属性页对话框包括向导对话框和一般属性页对话框两类,上一节鸡啄米讲了如何创建并显示向导对话框,本节将继续介绍一般属性页对话框的创建和显示。
实际上,一般属性页对话框的创建和显示过程和向导对话框是很类似的。鸡啄米将上一节中的向导对话框进行少量修改,使其成为一般属性页对话框。
一般属性页对话框的创建步骤:
1. 创建属性页对话框资源
属性页对话框资源的创建方法同向导对话框是一样的,上一讲中的对话框资源不需进行任何修改。
2. 创建属性页类
属性页类的创建和向导对话框的属性页类也基本一样,只是一般属性页对话框中不需要“下一步”和“完成”等按钮,所以上一讲中属性页类的OnSetActive和OnWizardFinish等重载函数可以去掉。即CSummandPage类中的OnSetActive函数、CAddPage类中的OnSetActive函数和OnWizardFinish函数可以删除或注释掉。其他部分不需作任何修改。
3. 创建属性表类
创建属性表类的过程同向导对话框属性表类也是一样的,所以上一讲中的CAddSheet类不需修改。
4. 显示一般属性页对话框
上一讲向导对话框的显示是在OnBnClickedInstructButton函数中实现的,其中语句sheet.SetWizardMode();旨在设置属性表为向导对话框模式,所以显示一般属性页对话框时不需调用SetWizardMode成员函数。另外,我们可以将属性页对话框的标题设为“使用说明”,在构造属性表对象时将此字符串作为构造函数的参数传入。OnBnClickedInstructButton函数修改如下:
-
void CAdditionDlg::OnBnClickedInstructButton()
-
{
-
// TODO: Add your control notification handler code here
-
-
// 创建属性表对象
-
CAddSheet sheet(_T("使用说明"));
-
-
// 打开模态一般属性页对话框
-
sheet.
DoModal();
-
}
这样一般属性页对话框的创建和显示就讲完了,我们运行下程序,在结果对话框上点“使用说明”按钮看看效果吧:
再总结下,一般属性页对话框和向导对话框的创建和显示的不同包括:是否需要OnSetActive和OnWizardFinish等重载函数,是否需要调用属性表类的SetWizardMode函数设置为向导对话框模式。
前面几节鸡啄米讲了属性页对话框,我们可以根据所讲内容方便的建立自己的属性页对话框。本节讲解Windows系统中最常用最简单的一类对话框--消息对话框。
我们在使用Windows系统的过程中经常会见到消息对话框,提示我们有异常发生或提出询问等。因为在软件开发中经常用到消息对话框,所以MFC提供了两个函数可以直接生成指定风格的消息对话框,而不需要我们在每次使用的时候都要去创建对话框资源和生成对话框类等。这两个函数就是CWnd类的成员函数MessageBox()和全局函数AfxMessageBox()。
下面鸡啄米就分别讲解两个函数的用法。
1 . CWnd::MessageBox()函数
CWnd::MessageBox()的函数原型如下:
-
int MessageBox(
-
LPCTSTR lpszText,
-
LPCTSTR lpszCaption =
NULL,
-
UINT nType = MB_OK
-
);
-
-
参数说明:
-
lpszText:需要显示的消息字符串。
-
lpszCaption:消息对话框的标题字符串。默认值为
NULL。取值为
NULL时使用默认标题。
-
nType:消息对话框的风格和属性。默认为 MB_OK 风格,即只有“确定”按钮。
-
nType 的取值可以是下面两个表中任取一个值,也可以是各取一个值的任意组合。
-
即可以指定一个对话框类型,也可以指定一个对话框图标,还可以两者都设定。
nType 的取值可以是下面两个表中任取一个值,也可以是各取一个值的任意组合。即可以指定一个对话框类型,也可以指定一个对话框图标,还可以两者都设定。下面表格是 对话框类型表
nType 取值 | 参数说明 |
MB_ABORTRETRY | 有“终止”、“重试”和“忽略”按钮 |
MB_OK | 有“确定”按钮 |
MB_OKCANCEL | 有“确定”和“取消”按钮 |
MB_RETRYCANCEL | 有“重试”和“取消”按钮 |
MB_YESNO | 有“是”和“否”按钮 |
MB_YESNOCANCEL | 有“是”、“否”和“取消”按钮 |
下面表格是 对话框图标表
nType 取值 | 显示图标 |
MB_ICONEXCLAMTION MB_ICONWARNING |
|
MB_ICONASTERISK MB_ICONINFORMATION |
|
MB_ICONQUESTION | |
MB_ICONHAND MB_ICONSTOP MB_ICONERROR |
如果想要设置nType的值为类型和图标的组合,可以像这样取值:MB_OKCANCEL | MB_ICONQUESTION。按位取或就可以了。
2 . AfxMessageBox()函数
AfxMessageBox()的函数原型为:
-
int AfxMessageBox(
-
LPCTSTR lpszText,
-
UINT nType = MB_OK,
-
UINT nIDHelp = 0
-
);
-
-
参数说明:
-
lpszText:同CWnd::
MessageBox()函数
-
nType:CWnd::
MessageBox()函数
-
nIDHelp:此消息的帮助的上下文ID。默认值为
0,取
0时表示要使用应用程序的默认帮助上下文。
我们在调用了上面两个函数后,都可以弹出模态消息对话框。消息对话框关闭后,我们也都可以得到它们的返回值。两者的返回值就是用户在消息对话框上单击的按钮的ID,可以是以下值:
IDABORT:单击“终止”按钮。
IDCANCEL:单击“取消”按钮。
IDIGNORE:单击“忽略”按钮。
IDNO:单击“否”按钮。
IDOK:单击“确定”按钮。
IDRETRY:单击“重试”按钮。
IDYES:单击“是”按钮。
我们还是拿前面加法计算器的程序做例子。
大家是否记得,在模态对话框及其弹出过程中我们修改了CAdditionDlg::OnBnClickedAddButton()函数,在点了“计算”按钮以后先弹出了一个模态对话框,询问用户是否确定要进行加法计算,并通过模态对话框DoModal函数的返回值判断用户选择了“确定”还是“取消”。这些功能很明显消息对话框完全能够实现,鸡啄米就使用消息对话框来替代原来的模态对话框。
在非模态对话框的创建及显示中,鸡啄米注释了模态对话框的相关代码,加入了非模态对话框的创建和显示代码,我们在加入消息对话框之前将非模态对话框的代码也注释或删除掉,确保此函数中不再生成原来的模态对话框或非模态对话框。
修改后的CAdditionDlg::OnBnClickedAddButton()函数如下:
-
void CAdditionDlg::OnBnClickedAddButton()
-
{
-
// TODO: Add your control notification handler code here
-
-
INT_PTR nRes;
-
-
// 显示消息对话框
-
nRes =
MessageBox(_T(
"您确定要进行加法计算吗?"), _T(
"加法计算器"), MB_OKCANCEL | MB_ICONQUESTION);
-
// 判断消息对话框返回值。如果为IDCANCEL就return,否则继续向下执行
-
if (IDCANCEL == nRes)
-
return;
-
-
// 将各控件中的数据保存到相应的变量
-
UpdateData(TRUE);
-
-
// 将被加数和加数的加和赋值给m_editSum
-
m_editSum = m_editSummand + m_editAddend;
-
-
// 根据各变量的值更新相应的控件。和的编辑框会显示m_editSum的值
-
UpdateData(FALSE);
-
// 设置属性对话框为向导对话框
-
//sheet.SetWizardMode();
-
}
编译运行,在运行结果对话框上点“计算”按钮弹出以下消息对话框:
大家也可以将MessageBox函数换为AfxMessageBox()函数,同时参数进行相应修改,运行下看看效果。
文件对话框的分类
文件对话框分为 打开文件对话框 和 保存文件对话框,相信大家在Windows系统中经常见到这两种文件对话框。例如,很多编辑软件像记事本等都有“打开”选项,选择“打开”后会弹出一个对话框,让我们选择要打开文件的路径,这个对话框就是打开文件对话框;除了“打开”选项一般还会有“另存为”选项,选择“另存为”后往往也会有一个对话框弹出,让我们选择保存路径,这就是保存文件对话框。
正如上面举例说明的,打开文件对话框用于选择要打开的文件的路径,保存文件对话框用来选择要保存的文件的路径。
文件对话框类CFileDialog
MFC使用文件对话框类CFileDialog封装了对文件对话框的操作。CFileDialog类的构造函数原型如下:
-
explicit CFileDialog(
-
BOOL bOpenFileDialog,
-
LPCTSTR lpszDefExt = NULL,
-
LPCTSTR lpszFileName = NULL,
-
DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
-
LPCTSTR lpszFilter = NULL,
-
CWnd* pParentWnd = NULL,
-
DWORD dwSize = 0,
-
BOOL bVistaStyle = TRUE
-
);
参数说明:
bOpenFileDialog:指定要创建的文件对话框的类型。设为TRUE将创建打开文件对话框,否则将创建保存文件对话框。
lpszDefExt:默认的文件扩展名。如果用户在文件名编辑框中没有输入扩展名,则由lpszDefExt指定的扩展名将被自动添加到文件名后。默认为NULL。
lpszFileName:文件名编辑框中显示的初始文件名。如果为NULL,则不显示初始文件名。
dwFlags:文件对话框的属性,可以是一个值也可以是多个值的组合。关于属性值的定义,可以在MSDN中查找结构体OPENFILENAME,元素Flags的说明中包含了所有属性值。默认为OFN_HIDEREADONLY和OFN_OVERWRITEPROMPT的组合,OFN_HIDEREADONLY表示隐藏文件对话框上的“Read Only”复选框,OFN_OVERWRITEPROMPT表示在保存文件对话框中如果你选择的文件存在了,就弹出一个消息对话框,要求确定是否要覆盖此文件。
lpszFilter:文件过滤器,它是由若干字符串对组成的一个字符串序列。如果指定了文件过滤器,则文件对话框中只有符合过滤条件的文件显示在文件列表中待选择。给大家看看VS2010 MSDN中给出的一个例子:
static TCHAR BASED_CODE szFilter[] = _T("Chart Files (*.xlc)|*.xlc|Worksheet Files (*.xls)|*.xls|Data Files (*.xlc;*.xls)|*.xlc; *.xls|All Files (*.*)|*.*||");
这样设置过滤器以后,文件对话框的扩展名组合框中将有四个选项:Chart Files (*.xlc)、Worksheet Files (*.xls)、Data Files(*.xlc;*.xls)和All Files (*.*),大家可以看到每种文件的扩展名规定都是一个字符串对,例如Chart Files的过滤字符串是Chart Files(*.xlc)和*.xlc成对出现的。
pParentWnd:文件对话框的父窗口的指针。
dwSize:OPENFILENAME结构体的大小。不同的操作系统对应不同的dwSize值。MFC通过此参数决定文件对话框的适当类型(例如,创建Windows 2000文件对话框还是XP文件对话框)。默认为0,表示MFC将根据程序运行的操作系统版本来决定使用哪种文件对话框。
bVistaStyle:指定文件对话框的风格,设为TRUE则使用Vista风格的文件对话框,否则使用旧版本的文件对话框。此参数仅在Windows Vista中编译时适用。
文件对话框也是模态对话框,所以在打开时也需要调用CFileDialog类的DoModal()成员函数。在打开文件对话框中点了“打开”或者在保存文件对话框中点了“保存”以后,我们可以使用CFileDialog类的成员函数GetPathName()获取选择的文件路径。
下面列出几个CFileDialog类的成员函数,我们可以使用它们获得文件对话框中的各种选择。
GetFileExt():获得选定文件的后缀名。
GetFileName():获得选定文件的名称,包括后缀名。
GetFileTitle():获得选定文件的标题,即不包括后缀名。
GetFolderPath():获得选定文件的目录。
GetNextPathName():获得下一个选定的文件的路径全名。
GetPathName():获得选定文件的路径全名。
GetReadOnlyPref():获得是否“以只读方式打开”。
GetStartPosition():获得文件名列表中的第一个元素的位置。
文件对话框实例
根据前面所讲内容,鸡啄米给大家做个文件对话框实例。
1.创建一个基于对话框的MFC应用程序工程,名称设为“Example17”。
2.修改主对话框IDD_EXAMPLE17_DIALOG的模板,删除自动生成的“TODO: Place dialog controls here.”静态文本框,添加两个编辑框,ID分别为IDC_OPEN_EDIT和IDC_SAVE_EDIT,再添加两个按钮,ID分别设为IDC_OPEN_BUTTON和IDC_SAVE_BUTTON,Caption分别设为“打开”和“保存”。按钮IDC_OPEN_BUTTON用于显示打开文件对话框,编辑框IDC_OPEN_EDIT显示在打开文件对话框中选择的文件路径。按钮IDC_SAVE_BUTTON用于显示保存文件对话框,编辑框IDC_SAVE_BUTTON显示在保存文件对话框中选择的文件路径。
3.分别为按钮IDC_OPEN_BUTTON和IDC_SAVE_BUTTON添加点击消息的消息处理函数CExample17Dlg::OnBnClickedOpenButton()和CExample17Dlg::OnBnClickedSaveButton()。
4.修改两个消息处理函数如下:
-
void CExample17Dlg::OnBnClickedOpenButton()
-
{
-
// TODO: Add your control notification handler code here
-
// 设置过滤器
-
TCHAR szFilter[] = _T(
"文本文件(*.txt)|*.txt|所有文件(*.*)|*.*||");
-
// 构造打开文件对话框
-
CFileDialog fileDlg(TRUE, _T("txt"), NULL, 0, szFilter, this);
-
CString strFilePath;
-
-
// 显示打开文件对话框
-
if (IDOK == fileDlg.
DoModal())
-
{
-
// 如果点击了文件对话框上的“打开”按钮,则将选择的文件路径显示到编辑框里
-
strFilePath = fileDlg.
GetPathName();
-
SetDlgItemText(IDC_OPEN_EDIT, strFilePath);
-
}
-
}
-
-
-
void CExample17Dlg::OnBnClickedSaveButton()
-
{
-
// TODO: Add your control notification handler code here
-
// 设置过滤器
-
TCHAR szFilter[] = _T(
"文本文件(*.txt)|*.txt|Word文件(*.doc)|*.doc|所有文件(*.*)|*.*||");
-
// 构造保存文件对话框
-
CFileDialog fileDlg(FALSE, _T("doc"), _T("my"), OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT, szFilter, this);
-
CString strFilePath;
-
-
// 显示保存文件对话框
-
if (IDOK == fileDlg.
DoModal())
-
{
-
// 如果点击了文件对话框上的“保存”按钮,则将选择的文件路径显示到编辑框里
-
strFilePath = fileDlg.
GetPathName();
-
SetDlgItemText(IDC_SAVE_EDIT, strFilePath);
-
}
-
}
上面显示编辑框内容时,鸡啄米使用了Windows API函数SetDlgItemText,当然也可以先给编辑框关联变量,然后再使用鸡啄米在创建对话框类和添加控件变量中介绍的
CDialogEx::UpdateData()函数,但是鸡啄米比较习惯使用SetDlgItemText函数,感觉比较灵活。
5.运行此程序,在结果对话框上点“打开”按钮,显示打开文件对话框如下:
点“保存”按钮后,显示保存文件对话框:
在打开文件对话框和保存文件对话框都选择了文件路径后,主对话框如下:
到此,文件对话框就讲完了,是不是依然很简单?如果忘记了文件对话框类构造函数的参数意义,可以回到鸡啄米来看看或者在MSDN上查阅。
字体对话框的作用是用来选择字体。我们也经常能够见到。MFC使用CFontDialog类封装了字体对话框的所有操作。字体对话框也是一种模态对话框。
CFontDialog类的构造函数
我们先来了解CFontDialog类。它的常用构造函数原型如下:
-
CFontDialog(
-
LPLOGFONT lplfInitial =
NULL,
-
DWORD dwFlags = CF_EFFECTS | CF_SCREENFONTS,
-
CDC* pdcPrinter =
NULL,
-
CWnd* pParentWnd =
NULL
-
);
-
-
参数说明:
-
lplfInitial:指向LOGFONT结构体数据的指针,可以通过它设置字体的一些特征。
-
dwFlags:指定选择字体的一个或多个属性,详情可在MSDN中查阅。
-
pdcPrinter:指向一个打印设备上下文的指针。
-
pParentWnd:指向字体对话框父窗口的指针。
上面的构造函数中第一个参数为LOGFONT指针,LOGFONT结构体中包含了字体的大部分特征,包括字体高度、宽度、方向、名称等等。下面是此结构体的定义:
-
type
def
struct tagLOGFONT {
-
LONG lfHeight;
-
LONG lfWidth;
-
LONG lfEscapement;
-
LONG lfOrientation;
-
LONG lfWeight;
-
BYTE lfItalic;
-
BYTE lfUnderline;
-
BYTE lfStrikeOut;
-
BYTE lfCharSet;
-
BYTE lfOutPrecision;
-
BYTE lfClipPrecision;
-
BYTE lfQuality;
-
BYTE lfPitchAndFamily;
-
TCHAR lfFaceName[LF_FACESIZE];
-
} LOGFONT;
获取字体对话框中所选字体
我们在字体对话框中选择了字体后,如何获取选定的字体呢?我们可以通过CFontDialog类的成员变量m_cf间接获得选定字体的CFont对象。m_cf是CHOOSEFONT类型的变量,CHOOSEFONT结构体定义如下:
-
typedef struct {
-
DWORD lStructSize;
-
HWND hwndOwner;
-
HDC hDC;
-
LPLOGFONT lpLogFont;
-
INT iPointSize;
-
DWORD Flags;
-
COLORREF rgbColors;
-
LPARAM lCustData;
-
LPCFHOOKPROC lpfnHook;
-
LPCTSTR lpTemplateName;
-
HINSTANCE hInstance;
-
LPTSTR lpszStyle;
-
WORD nFontType;
-
INT nSizeMin;
-
INT nSizeMax;
-
} CHOOSEFONT, *LPCHOOSEFONT;
CHOOSEFON结构体中有个成员lpLogFont,它是指向LOGFONT结构体变量的指针,就像上面所说,LOGFONT中包含了字体特征,例如,我们可以通过LOGFONT的lfFaceName得知字体名。
我们最终要获得的是所选择字体的CFont对象,有了字体的LOGFONT怎样获得对应的CFont对象呢?使用CFont类的成员函数CreateFontIndirect可以达到此目的。函数原型如下:
BOOL CreateFontIndirect(const LOGFONT* lpLogFont );
参数是LOGFONT指针类型,我们可以传入CFontDialog类成员变量m_cf的lpLogFont成员,就可以得到所选字体的CFont对象了。
字体对话框应用实例
鸡啄米给大家做一个字体对话框的实例。先介绍此实例要实现的功能,生成一个对话框,对话框中放置一个“字体选择”按钮和一个编辑框。点击“字体选择”按钮将弹出字体对话框。编辑框用于显示所选字体名,并以选定的字体来显示字体名字符串,例如,如果选择了宋体,则在编辑框中以宋体显示字符串“宋体”。
以下是创建此实例的步骤:
1.创建一个基于对话框的MFC工程,名字为“Example18”。
2.在自动生成的主对话框IDD_EXAMPLE18_DIALOG的模板中,删除“TODO: Place dialog controls here.”静态文本框,添加一个按钮,ID设为IDC_FONT_BUTTON,Caption设为“字体选择”,用于显示字体对话框来选择字体,再添加一个编辑框,ID设为IDC_FONT_EDIT,用来以所选字体显示字体名字符串。
3.在Example18Dlg.h中为CExample18Dlg类添加private成员变量:CFont m_font;,用来保存编辑框中选择的字体。
4.为按钮IDC_FONT_BUTTON添加点击消息的消息处理函数CExample18Dlg::OnBnClickedFontButton()。
5.修改消息处理函数CExample18Dlg::OnBnClickedFontButton()如下:
-
void CExample18Dlg::OnBnClickedFontButton()
-
{
-
// TODO: Add your control notification handler code here
-
CString strFontName;
// 字体名称
-
LOGFONT lf;
// LOGFONT变量
-
-
// 将lf所有字节清零
-
memset(&lf,
0,
sizeof(LOGFONT));
-
-
// 将lf中的元素字体名设为“宋体”
-
_tcscpy_s(lf.lfFaceName, LF_FACESIZE, _T(
"宋体"));
-
-
// 构造字体对话框,初始选择字体名为“宋体”
-
CFontDialog fontDlg(&lf);
-
-
if (IDOK == fontDlg.
DoModal())
// 显示字体对话框
-
{
-
// 如果m_font已经关联了一个字体资源对象,则释放它
-
if (m_font.m_hObject)
-
{
-
m_font.
DeleteObject();
-
}
-
// 使用选定字体的LOGFONT创建新的字体
-
m_font.
CreateFontIndirect(fontDlg.m_cf.lpLogFont);
-
// 获取编辑框IDC_FONT_EDIT的CWnd指针,并设置其字体
-
GetDlgItem(IDC_FONT_EDIT)->
SetFont(&m_font);
-
-
// 如果用户选择了字体对话框的OK按钮,则获取被选择字体的名称并显示到编辑框里
-
strFontName = fontDlg.m_cf.lpLogFont->lfFaceName;
-
SetDlgItemText(IDC_FONT_EDIT, strFontName);
-
}
-
}
6.最后,编译运行程序。显示结果对话框,点击“字体选择”按钮,将弹出字体对话框,默认选择为“宋体”,我们改而选择“华文彩云”字体点“确定”,编辑框中会像如下显示:
到此,我们又学会了字体对话框的使用,对于以后在界面开发中控制显示的字体很有帮助。有问题欢迎在鸡啄米留言。
颜色对话框大家肯定也不陌生,我们可以打开它选择需要的颜色,简单说,它的作用就是用来选择颜色。MFC中提供了CColorDialog类封装了颜色对话框的所有操作,我们可以通过它显示颜色对话框,并获取颜色对话框中选择的颜色。颜色对话框跟字体对话框一样,也是一种模态对话框。
CColorDialog类的构造函数
CColorDialog(
COLORREF clrInit = 0,
DWORD dwFlags = 0,
CWnd* pParentWnd = NULL
);
参数说明:
clrInit:默认选择颜色的颜色值,类型为COLORREF,实际上就是unsigned long类型。如果没有设置它的值,则默认为RGB(0,0,0),即黑色。
注:RGB(r,g,b)是宏,可以计算颜色值。括号中的三个值分别为红、绿、蓝分量的值。
dwFlags:自定义颜色对话框功能和外观的属性值。详情可在MSDN中查阅。
pParentWnd:颜色对话框的父窗口的指针。
获取颜色对话框中所选颜色值
我们使用颜色对话框的最终目的还是要获得在颜色对话框中选择的颜色值。为此CColorDialog类的成员函数GetColor()能够很好的实现我们的要求。GetColor()函数的原型为:
COLORREF GetColor( ) const;
它返回所选颜色的COLORREF值。
如果我们想获得R、G、B各分量的值呢?可以根据GetColor得到的COLORREF颜色值,通过使用GetRValue、GetGValue和GetBValue三个宏获得。GetRValue的语法形式为:
BYTE GetRValue(DWORD rgb);
参数rgb就是COLORREF颜色值,返回值即是R分量值。其他两个宏的形式与之类似。例如,GetColor()函数返回的COLORREF为10000,则R分量值就是GetRValue(10000)。
颜色对话框应用实例
鸡啄米下面给大家做一个颜色对话框的小例子。此例要实现的功能简单介绍下:生成一个对话框,对话框中放置一个“颜色选择”按钮,四个静态文本框和四个编辑框。四个静态文本框分别显示Color:、R:、G:、B:,每个静态文本框后面跟一个编辑框,分别用来显示颜色对话框中选择的颜色值和所选颜色值的红色分量、绿色分量、蓝色分量。
以下是实例创建的步骤:
1.创建一个基于对话框的MFC工程,名字为“Example19”。
2.在自动生成的主对话框IDD_EXAMPLE19_DIALOG的模板中,删除“TODO: Place dialog controls here.”静态文本框,添加一个按钮,ID设为IDC_COLOR_BUTTON,Caption设为“颜色选择”,用于显示颜色对话框来选择颜色。再添加四个静态文本框,ID分别为IDC_COLOR_STATIC、IDC_R_STATIC、IDC_G_STATIC、IDC_B_STATIC,Caption分别设为“Color:”、“R:”、“G:”、“B:”,然后每个静态文本框后添加一个编辑框,四个编辑框的ID分别为IDC_COLOR_EDIT、IDC_R_EDIT、IDC_G_EDIT、IDC_B_EDIT,分别用来显示颜色对话框中选择的颜色值和所选颜色值的红色分量、绿色分量、蓝色分量。
3.为按钮IDC_COLOR_BUTTON添加点击消息的消息处理函数CExample19Dlg::OnBnClickedColorButton()。
4.修改消息处理函数CExample19Dlg::OnBnClickedColorButton()如下:
-
void CExample19Dlg::OnBnClickedColorButton()
-
{
-
// TODO: Add your control notification handler code here
-
COLORREF color =
RGB(
255,
0,
0);
// 颜色对话框的初始颜色为红色
-
CColorDialog colorDlg(color);
// 构造颜色对话框,传入初始颜色值
-
-
if (IDOK == colorDlg.
DoModal())
// 显示颜色对话框,并判断是否点击了“确定”
-
{
-
color = colorDlg.
GetColor();
// 获取颜色对话框中选择的颜色值
-
SetDlgItemInt(IDC_COLOR_EDIT, color);
// 在Color编辑框中显示所选颜色值
-
SetDlgItemInt(IDC_R_EDIT,
GetRValue(color));
// 在R编辑框中显示所选颜色的R分量值
-
SetDlgItemInt(IDC_G_EDIT,
GetGValue(color));
// 在G编辑框中显示所选颜色的G分量值
-
SetDlgItemInt(IDC_B_EDIT,
GetBValue(color));
// 在B编辑框中显示所选颜色的B分量值
-
}
-
}
5.最后编译运行程序,在结果对话框中点击“颜色选择”按钮,弹出颜色对话框。初始状态下,选择框在红色上,我们选另一种颜色,此时的颜色对话框如下:
点“确定”,主对话框上的四个编辑框中分别显示了选择的颜色值、R分量、G分量和B分量: