摘要:本文将介绍如何在可停靠窗口(Dockable Pane)中使用对话框来来实现可视化设计,即将一个对话框(Dialog)作为子窗口填充在可停靠窗格之中,这样做的好处是使得可以通过Visual Studio的对话框资源编辑功能可视化地设计窗口,并轻松地实现控件的消息处理程序。

关键字:Dockable Pane, Dialog, 可视化设计

 

一、使用可停靠窗格开发用户界面

很多程序中都使用可停靠窗格作为用户界面中的重要组成部分,最熟悉的例子莫过于使用最多的Visual Studio。从VS2008 SP1和VS2010开始,新版的MFC框架中已经提供了一系列的类实现这样的功能,其中最值得关注的是CDockablePane类。

CDockablePane类代表了一个可停靠窗格,其用法可以通过阅读Visual Studio的“MFC应用程序向导”自动生成的源代码来习得。创建一个可停靠窗格的步骤大致分为五步:

(1) 定义一个类继承自CDockablePane以实现特定的功能。

(2) 在CMainFrame中定义上述类型成员变量。

(3) 在CMainFrame的OnCreate函数中调用CDockablePane的Create函数创建窗格。

(4) 调用CDockablePane的EnableDocking函数配置可停靠位置。

(5) 调用CMainFrame的DockPane函数停靠此窗格。

详情可参考Visual Studio生成的代码,此处不再赘述。

美中不足的是,Visual Studio并未提供对可停靠窗格进行可视化设计的支持。不过,如果需要,可以通过对话框的功能来间接实现。

二、设计思路

依照使用方便、代码可重用的设计原则,有以下几个方面需要考虑:

(1) 应当提供用户一个类来继承使用(CDockableForm),为了设计在对话框编辑视图中添加控件关联变量和事件处理程序,CDockableForm类应该是CDialog类的子类。

(2) 用户创建此类的对象时,CDockablePane类的对象也应该随之创建。该类提供一个Create函数,在该函数中应该同时完成可停靠窗格和对话框的创建过程。

(3) 对话框应该铺满可停靠窗格,并随可停靠窗格隐藏而隐藏、显示而显示。这需要将对话框设置为可停靠窗格的子窗口,并添加一个类(CDockablePaneAsContainer)继承自CDockablePane,在其WM_SIZE消息处理函数中调整对话框的位置。

(4) 在可停靠窗格销毁时,应该同时销毁对话框。可以可停靠窗格的WM_DESTROY消息处理函数中销毁对话框。

(5) CDockablePane类应该提供一个成员函数使得用户可以访问其CDockablePane成员,以实现其在主框架窗口(CMainFrame)中的停靠功能。

依上所述,设计的类如图所示。

在可停靠窗格中使用对话框来实现可视化设计_Visual Studio

图2-1 类图

CDockableForm类和CDockablePaneAsContainter类的实现代码见附录1-4。

使用方法分为四步:

(1) 创建、编辑对话框资源,添加对话框类,基类选择CDialog类。

(2) 在项目中添加DockableForm.h和DockableForm.cpp(见附录5),在对话框类的头文件中包含DockableForm.h,将对话框类的基类改为CDockableForm,将对话框类的头文件和源代码文件中所有CDialog替换为CDockableForm,同时也要修改一个对话框类的构造函数,因为CDockableForm类的构造函数与CDialog类的不一样。

(3) 在CMainFrame类中添加第二步生成的类的成员,在其OnCreate函数中调用该成员的Create函数创建可停靠窗格,调用GetDockablePane方法得到其可停靠窗格的引用以实现停靠功能。

使用示例见附录5。

附录

1. CDockableForm类的声明代码

class CDockablePaneAsContainer : public CDockablePane
{
public:
    CDockablePaneAsContainer(CDialog* pDialog) : m_pDialog(pDialog) { }
 
private:
    CDialog* m_pDialog;
 
public:
    DECLARE_MESSAGE_MAP()
    afx_msg void OnSize(UINT nType, int cx, int cy);
    afx_msg void OnDestroy();
};

 

2. CDockableForm的实现代码。

BEGIN_MESSAGE_MAP(CDockablePaneAsContainer, CDockablePane)
    ON_WM_SIZE()
    ON_WM_DESTROY()
END_MESSAGE_MAP()
 
void CDockablePaneAsContainer::OnSize(UINT nType, int cx, int cy)
{
    CDockablePane::OnSize(nType, cx, cy);
     
    // TODO: 在此处添加消息处理程序代码
    if (m_pDialog->GetSafeHwnd())
    {
        CRect rc;
        GetClientRect(rc);
        m_pDialog->MoveWindow(rc);
    }
}
 
void CDockablePaneAsContainer::OnDestroy()
{
    CDockablePane::OnDestroy();
 
    // TODO: 在此处添加消息处理程序代码
    m_pDialog->DestroyWindow();
}

3. CDockableForm类的声明代码

class CDockableForm : public CDialog
{
public:
    CDockableForm(UINT nIDTemplate);
     virtual BOOL Create(
    LPCTSTR lpszCaption,
         CWnd* pParentWnd,
        const RECT& rect,
        BOOL bHasGripper,
        UINT nID, 
        DWORD dwStyle,
        DWORD dwTabbedStyle = AFX_CBRS_REGULAR_TABS,
        DWORD dwControlBarStyle = AFX_DEFAULT_DOCKING_PANE_STYLE,
        CCreateContext* pContext = NULL);
    CDockablePane& GetDockablePane() { return m_wndPane; }
private:
    CDockablePaneAsContainer m_wndPane;
};

 

4. CDockablePaneAsContainer的实现代码

CDockableForm::CDockableForm(UINT nIDTemplate)
: CDialog(nIDTemplate, &m_wndPane)
, m_wndPane(this)
{
}
 
BOOL CDockableForm::Create(LPCTSTR lpszCaption, CWnd *pParentWnd, const RECT &rect, BOOL bHasGripper, UINT nID, DWORD dwStyle, DWORD dwTabbedStyle, DWORD dwControlBarStyle, CCreateContext *pContext)
{
    m_wndPane.Create(lpszCaption, pParentWnd, rect, bHasGripper, nID, dwStyle, dwTabbedStyle, dwControlBarStyle, pContext);
    CDialog::Create(m_nIDHelp, &m_wndPane);
    SetParent(&m_wndPane);
    ShowWindow(SW_SHOW);
    return TRUE;
}

 

5. 示例代码