在CDockablePane中嵌入CFormView

时间:2023-03-09 07:25:32
在CDockablePane中嵌入CFormView

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什么时候销毁?