一、控件的介绍
MFC中封装了许多有用的类,CTabCtrl和CPropertySheet是其中两个比较常用的类型。CTabCtrl即Tab控件对应的类,Windows应用程序中大量使用了Tab控件。属性页对应的是CPropertySheet,你在桌面上右键鼠标弹出的显示对话框就是一个CPropertySheet类型的对象。事实上这类弹出的、类似“属性页”的窗口大部分都是CPropertySheet。
其实,这两个类之间也有着很密切的联系。使用Spy++观察一个属性页窗口可以发现,它也包含了一个Tab控件
二、属性页的使用
属性页的使用比较简单,过程也很清晰。先说一下,CPropertySheet和CDialog对话框类同样都派生于CWnd类,所以他俩在使用上有很多相似的地方。我不知道是不是这哥俩太相似了,以至于Spy++把我自建的一个CPropertySheet窗口认作是CDialog。
既然属性页和对话框相似,我们只要有了一个属性页的类,就可以使用DoModal()方法来产生一个模态的窗口,或者用Create()产生一个非模态的窗口。注意如果显示非模式的窗口,要考虑变量生存期的问题,这点和非模态对话框是一样的。
属性页窗口里面那么多的选项卡是怎么弄出来的?这就需要CPropertyPage。这是一个对话框类的子类,对应着一个选项卡。对一个属性页,我们使用AddPage()方法来添加选项卡
选项卡既然是对话框,我们就可以在资源窗口中建立对话框,然后利用类向导为他建立类,注意要选择派生自CPropertyPage。MSDN上面说,建立这类对话框资源模板时有几点要注意
1.保留TitleBar属性。对话框标题就是将来显示在选项卡上的标签
2.对话框风格设为Child,边框设为Thin
3.将对话框设为Disabled
好了,总结一下属性页的使用步骤就是
1.建立模板资源,构造你需要的每一个CPropertyPage。
2.使用CPropertySheet类,用AddPage()方法为其添加至少一个选项卡。或者,你可以派生一个自己的CMyPropertySheet类,在内部添加上各个CPropertyPage作为成员变量,并在构造函数中用AddPage
3.在需要使用属性页的地方声明变量,使用DoModal()或者Create()方法显示属性页
三、Tab控件的使用
Tab控件一般用在对话框中。在我看来Tab控件本身并不复杂,其实也就是一组简单的按钮而已,因为它并不能直接对各个选项卡进行操纵。我们需要对用户选择选项卡的行为进行手工地响应
比如你希望对每一个选项卡显示不同的内容,你必须通过显示/隐藏不需要的内容。当每一个选项卡上的内容较多时,可以把这些内容放在一个容器里,比如一个对话框。这样,一个选项卡对应一个对话框,关系比较清晰。这些对话框应设置为没有标题栏、Child风格、无边框的、非模式的。以此为例,对话框中使用Tab控件的步骤是
1.在对话框中添加Tab控件
2.为各选项卡建立对话框模板资源
3.响应Tab控件的TCN_SELCHANGE消息,在OnSelChange()函数中,用ShowWindow()函数显示欲显示的选项卡,隐藏其他。非模式对话框的生存期问题这里仍然存在,不可忽视
4.初始化Tab控件。可在对话框初始化时手工调用OnSelChange()函数的方法来实现.
下面使用使用CTabCtrl或CPropertySheet实现标签页:
CTabCtrl
1.先建立一个MFC应用程序,然后在资源管理器中新建一对话框,并为它建立对应的类,然后从工具箱里面把Tab Control控件放入该对话框中。
再在对话框类中,声明一个CTabCtrl变量:
CTabCtrl m_tab;
变量m_tab用来与对话框中的Tab Control控件交互,为此要在DoDataExchange函数中加入DDX_Control语句:
DDX_Control(pDX, IDC_TAB_NEW, m_tab);
IDC_TAB_NEW是Tab Control控件的ID。
2.建立两个新对话框,用来当做Tab Control控件的两个页。别忘了把Style改为Child,Border改为None。然后就可以在上面加其他控件了。
接着分别为这两个对话框建立两个类,比如CPage1和CPage2。
然后在对话框类头文件中,加入这两个对话框对象。同时增加一个变量int m_CurSelTab,用了表明是哪个Page即将被切换。
int m_CurSelTab;
CPage1 m_page1;
CPage2 m_page2;
CDialog* pDialog[2]; //用来保存对话框对象指针
在对话框类的初始化函数中需要把CPage1、CPage2和Tab Control关联起来,并保存页面地址,设置初始页面。
//为Tab Control增加两个页面 m_tab.InsertItem(0, _T("Farm")); m_tab.InsertItem(1, _T("Note")); //创建两个对话框 m_page1.Create(IDD_DIALOG1, &m_tab); m_page2.Create(IDD_DIALOG2, &m_tab); //设定在Tab内显示的范围 CRect rc; m_tab.GetClientRect(rc); rc.top += 20; m_page1.MoveWindow(&rc); m_page2.MoveWindow(&rc); //把对话框对象指针保存起来 pDialog[0] = &m_page1; pDialog[1] = &m_page2; //显示初始页面 pDialog[0]->ShowWindow(SW_SHOW); pDialog[1]->ShowWindow(SW_HIDE); //保存当前选择 m_CurSelTab = 0;
这里我们用了一个CDialog指针数组来进行保存Tab Control的每个页,数组的大小是Tab Control页面的个数,数组下标对应着每个页面的索引(这样方便快速存取)。接下来,为Tab Control添加消息处理程序:
ON_NOTIFY(TCN_SELCHANGE, IDC_TAB_NEW, &CTabDlg::OnTcnSelchangeTabNew)
void CTabDlg::OnTcnSelchangeTabNew(NMHDR *pNMHDR, LRESULT *pResult) { //把当前的页面隐藏起来 pDialog[m_CurSelTab]->ShowWindow(SW_HIDE); //得到新的页面索引 m_CurSelTab = m_tab.GetCurSel(); //把新的页面显示出来 pDialog[m_CurSelTab]->ShowWindow(SW_SHOW); *pResult = 0; }
3.如果要在Tab Control控件外面获取其内部对话框中控件的数据,就需要使用DDX/DDV机制,还要调用相应的UpdateData函数。
m_page1.UpdateData();
m_page2.UpdateData();
CPropertySheet
1.先建立MFC应用程序,然后在资源管理器中新建一对话框,并为它建立对应的类,在编辑该对话框,可以自由加一些所需的控件,但得留出一定的空间用于放至
标签页。在主对话框类里加入一个CPropertySheet类的一个成员变量m_sheet代表整个标签页。
2.接着分别为创建两个对话框,建立两个与之对应的类时,注意选择基类为CPropertyPage。假设新生成两个类为CPage1,CPage2。
3.在刚才加入m_sheet成员的位置,加入上述类型成员变量m_page1,m_page2。
CPropertySheet m_sheet;
CPage1 m_page1;
Cpage2 m_page2;
在主对话框类的OnInitDialog()函数中加入:
//加入标签,标签名由各个子对话框的标题栏决定 m_sheet.AddPage(&m_page1); m_sheet.AddPage(&m_page2); //对修改标签页title m_page1.m_psp.dwFlags|=PSP_USETITLE; m_page1.m_psp.pszTitle= "标签页A"; m_page2.m_psp.dwFlags|=PSP_USETITLE; m_page2.m_psp.pszTitle= "标签页B"; //用Create来创建一个属性页 m_sheet.Create(this, WS_CHILD | WS_VISIBLE, WS_EX_CONTROLPARENT); //调整m_sheet位置 RECT rect; GetWindowRect(&rect); m_sheet.MoveWindow(&rect);