CDockablePane中嵌入CFormView与嵌入CDialogEx稍有不同,差异主要体现在CFormView类本身与CDialogEx类的不同上,CDockablePane层面的操作完全相同。
a) 创建单文档应用程序;
b) 加入对话框资源,注意,对话框必须有Child属性,Border设置为None
由CFormView派生的类,可以关联一个对话框资源。但该对话框资源必须在属性设定中Style选定[Child]属性,否则的话,
代码可以编译,但Debug运行会报告一个断言错误,跟踪代码,断言在:
#ifdef _DEBUG
// dialog template
must exist and be invisible with WS_CHILD set
if (!_AfxCheckDialogTemplate(m_lpszTemplateName, TRUE))
{
ASSERT(FALSE); //
invalid dialog template name
PostNcDestroy(); // cleanup if
Create fails too soon
return FALSE;
}
#endif //_DEBUG
CFormView比较特殊,是一个父窗体嵌套了一个子窗体,所以CFormView类的派生类的实例不响应WM_CLOSE消息,仅仅响应WM_DESTROY消息。另外,若要用代码关闭当前View,也不能直接:PostMessage(WM_CLOSE,0,0);而必须先获取父窗体的指针,然后对父窗体发送WM_CLOSE消息才行,像这样:GetParent()->PostMessage(WM_CLOSE,0,0);才能够达到目的。《深入浅出MFC》第八章461页图8-1清楚地说明了这种情况,View窗口是CChildFrame窗口的子窗口。
c) 为对话框创建类CFormViewEmbeded,基类为CFormView
d) 重载CFormViewEmbeded类的Create函数,将访问权限改为Public(基类为protect),不用添加额外代码,为了在程序中使用该函数
e) 重写CFormViewEmbeded类的OnMouseActive消息响应函数,为了防止CDockablePane处于悬浮状态时程序崩溃(不重载必然崩溃!)
int CFormViewEmbeded::OnMouseActivate(CWnd* pDesktopWnd, UINT nHitTest, UINT message)
{
// TODO: 在此添消息处理程序代码
int nResult = 0;
CFrameWnd* pParentFrame = GetParentFrame();
if( pParentFrame == pDesktopWnd )
{
// When this is docked
nResult= CFormView::OnMouseActivate(pDesktopWnd, nHitTest, message);
}
else
{
// When this is not docked
BOOL isMiniFrameWnd =pDesktopWnd->IsKindOf( RUNTIME_CLASS( CMiniFrameWnd ) );
BOOL isPaneFrameWnd =pDesktopWnd->IsKindOf( RUNTIME_CLASS( CPaneFrameWnd ) );
BOOL isMultiPaneFrameWnd =
pDesktopWnd->IsKindOf( RUNTIME_CLASS( CMultiPaneFrameWnd ) );
// pDesktopWnd is the frame window for CDockablePane
nResult = CWnd::OnMouseActivate( pDesktopWnd, nHitTest, message );
}
return nResult;
}
f) 创建派生自CDockablePane的类CDockableFormView
g) 由于CFormView类的构造函数访问权限为protect,不能直接声明变量,所以为CDockableFormView添加CDialogEmbeded*类型的成员变量m_pFormViewEmbeded,(另一种解决方法是将CMainFrame声明为友元类);
h) 在CDockableFormView类的构造函数中添加
m_pFormViewEmbeded =
(CFormViewEmbeded*)(RUNTIME_CLASS(CFormViewEmbeded))->CreateObject();
i) 重载CDockableFormView的OnCreate函数
int CDockableFormView::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CDockablePane::OnCreate(lpCreateStruct) == -1)
return -1;
// TODO: 在此添加您专用的代码
CRect rect;
GetClientRect(&rect);
m_pFormViewEmbeded->Create(NULL,NULL,WS_CHILD|WS_VISIBLE,rect,this,0,NULL);
return 0;
}
a) 重载CDockableFormView的OnSize函数
void CDockableFormView::OnSize(UINT nType, int cx, int cy)
{
CDockablePane::OnSize(nType, cx, cy);
// TODO: 在此处添加消息处理程序代码
if(m_pFormViewEmbeded->GetSafeHwnd())
{
CRect rect;
GetClientRect(&rect);
m_pFormViewEmbeded->SetWindowPos(NULL,
rect.left,rect.top,rect.Width(),rect.Height(),
SWP_NOACTIVATE|SWP_NOZORDER);
}
}
b) 重载CDockableFormView的OnDestory函数
void CDockableFormView::OnDestroy()
{
CDockablePane::OnDestroy();
// TODO: 在此处添加消息处理程序代码
m_pFormViewEmbeded->DestroyWindow();
}
c) 在框架类中添加CDockableDlg对象m_dockDlg
d) 在CMainFrame类的OnCreate函数中添加以下代码,Create函数里面的1002是这个停靠栏的ID,这里是随便指定的一个数值,只要不和其他已用资源重复即可,真正应用的时候,以在字符串表中添加一个ID
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
…
EnableDocking(CBRS_ALIGN_ANY);
m_dockDlg.Create("Dock FormView",this,CRect(0,0,200,200),TRUE,1002,
WS_CHILD|WS_VISIBLE|WS_CLIPSIBLINGS|WS_CLIPCHILDREN|CBRS_RIGHT|CBRS_FLOAT_MULTI);
m_dockDlg.EnableDocking(CBRS_ALIGN_ANY);
DockPane(&m_dockFormView);
return 0;
}
e) 疑问:m_dockFormView什么时候销毁?