MFC Cdockablepane AttachToTabWnd 選項卡 拖拽 點擊 雙擊 標籤欄 子停靠面板消失問題解決方案

       具體的MFC停靠面板的佈局問題在這裏我就不說了,網上已經有比較成熟的教程,大神也給出了詳細可靠的教程,鏈接如下:

http://blog.csdn.net/chenlycly/article/details/38964113

http://www.xuebuyuan.com/2089010.html


      現在的我所做的項目要解決的問題是:設置五個停靠面板a,b,c,d,e依次放在同一個選項卡O裏面,這個選項卡O可以停靠在邊界(即具有專門的m_pDefaultSlider對View進行分割),拖拽分割線改變寬度,自動隱藏,但是O不可以浮動(floating),同時O裏面的五個停靠面板同樣不可以浮動,只能固定在選項卡O中;

經過一兩天的嘗試,最終做出了比較滿意的效果,如下:



如圖,所有的停靠面板依次停靠在選項卡中,選項卡沒有關閉按鈕,同時五個停靠面板也沒有關閉按鈕,代碼如下:


CDockablePane* m_pTabbedBar;//五個子面板的選項卡

//創建面板

BOOL CMainFrame::CreateDockingWindows()
{
BOOL bNameValid;

CString strDesignInfoPane;
bNameValid = strDesignInfoPane.LoadString(IDS_A_PANE);
ASSERT(bNameValid);

//a面板
if (a)
{
if (!a->Create(strDesignInfoPane, this, CRect(0, 0, 400, 800), TRUE, IDS_A_PANE, WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | CBRS_LEFT/* | CBRS_FLOAT_MULTI*/ /*| CBRS_SIZE_FIXED*/, AFX_CBRS_AUTOHIDE))
{
TRACE0("未能創建“a”窗口\n");
return FALSE; // 未能創建
}
a->SetMinSize(CSize(400, 800));
}

bNameValid = strDesignInfoPane.LoadString(IDS_B_PANE);
ASSERT(bNameValid);
//b面板
if (b)
{
if (!b->Create(strDesignInfoPane, this, CRect(0, 0, 400, 800), TRUE, IDS_B_PANE, WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | CBRS_LEFT/* | CBRS_FLOAT_MULTI*/ /*| CBRS_SIZE_FIXED*/, AFX_CBRS_AUTOHIDE))
{
TRACE0("未能創建“b”窗口\n");
return FALSE; // 未能創建
}
b->SetMinSize(CSize(400, 800));
}


//c面板
CString strParameterInfoPane;
bNameValid = strParameterInfoPane.LoadString(IDS_C_PANE);
ASSERT(bNameValid);


if (c)
{
if (!c->Create(strParameterInfoPane, this, CRect(0, 0, 400, 800), TRUE, IDS_C_INFO_PANE, WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | CBRS_LEFT | CBRS_FLOAT_MULTI /*| CBRS_SIZE_FIXED*/, AFX_CBRS_AUTOHIDE))
{
TRACE0("未能創建“c”窗口\n");
return FALSE; // 未能創建
}
c->SetMinSize(CSize(400, 800));
}


//創建d面板
CString strStatusInfoPane;
bNameValid = strStatusInfoPane.LoadString(IDS_D_PANE);
ASSERT(bNameValid);


if (d)
{
if (!d->Create(strStatusInfoPane, this, CRect(0, 0, 400, 800), TRUE, IDS_D_PANE, WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | CBRS_LEFT | CBRS_FLOAT_MULTI /*| CBRS_SIZE_FIXED*/ ,AFX_CBRS_AUTOHIDE))
{
TRACE0("未能創建“d”窗口\n");
return FALSE; // 未能創建
}
d->SetMinSize(CSize(400, 800));
}




//創建e面板
bNameValid = strStatusInfoPane.LoadString(IDS_E_PANE);
ASSERT(bNameValid);


if (e)
{
if (!e->Create(strStatusInfoPane, this, CRect(0, 0, 400, 800), TRUE, IDS_E_PANE, WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | CBRS_LEFT | CBRS_FLOAT_MULTI /*| CBRS_SIZE_FIXED*/, AFX_CBRS_AUTOHIDE))
{
TRACE0("未能創建“e”窗口\n");
return FALSE; // 未能創建
}
e->SetMinSize(CSize(400, 800));
}

RecalcLayout();
return TRUE;
}

//設置面板佈局,放在同一個選項卡里
BOOL CMainFrame::SetDockingWindowsLayout()
{

if (a)
{
a->EnableDocking(CBRS_ALIGN_ANY);
DockPane(designinfopane);
//a->SetControlBarStyle(~AFX_CBRS_CLOSE);//去除關閉按鈕
//a->SetControlBarStyle(AFX_CBRS_RESIZE);//隱藏後,再顯示沒有可隱藏按鈕
a->SetControlBarStyle(AFX_CBRS_AUTOHIDE);
a->DockToWindow(designinfopane, CBRS_ALIGN_TOP);
}
else
{
return FALSE;
}


if (b)
{
b->EnableDocking(/*CBRS_ALIGN_LEFT*/CBRS_ALIGN_ANY);
b->SetControlBarStyle(AFX_CBRS_AUTOHIDE);
b->DockToWindow(a, CBRS_RIGHT);
b->AttachToTabWnd(a, DM_SHOW, FALSE, &m_pTabbedBar);
a->DockToWindow(b, CBRS_ALIGN_TOP);


}
else
{
return FALSE;
}


if (c)
{
c->EnableDocking(/*CBRS_ALIGN_LEFT*/CBRS_ALIGN_ANY);
c->SetControlBarStyle(AFX_CBRS_AUTOHIDE);
c->DockToWindow(b, CBRS_RIGHT);
c->AttachToTabWnd(b, DM_SHOW, FALSE, &m_pTabbedBar);
}
else
{
return FALSE;
}


if (d)
{
d->EnableDocking(/*CBRS_ALIGN_LEFT*/CBRS_ALIGN_ANY);
d->SetControlBarStyle(AFX_CBRS_AUTOHIDE);
d->DockToWindow(c, CBRS_RIGHT);
d->AttachToTabWnd(c, DM_SHOW, FALSE, &m_pTabbedBar);
}
else
{
return FALSE;
}

        if (e)
{
e->EnableDocking(/*CBRS_ALIGN_LEFT*/CBRS_ALIGN_ANY);
e->SetControlBarStyle(AFX_CBRS_AUTOHIDE);
e->DockToWindow(d, CBRS_RIGHT);
e->AttachToTabWnd(d, DM_SHOW, FALSE, &m_pTabbedBar);
}
else
{
return FALSE;
}




if (m_pTabbedBar)
{
m_pTabbedBar->SetControlBarStyle(AFX_CBRS_AUTOHIDE);//去除關閉按鈕,同時設置可以自動隱藏,又不可以浮動
}

return TRUE;
}


現在保留自動隱藏,又禁止面板浮動的功能完成了,但是出現一個問題就是:五個子面板雙擊或拖動其在選項卡中的標籤按鈕會出現消失的現象,感覺MFC這個功能給的接口也不完善。從網上查了很多資料都沒有解決方案。最後,我想了一個變通的方法,成功實現了不使子面板消失的功能,代碼如下:

重載以下方法:

// override command routing 
virtual BOOL OnCmdMsg(UINT id, int code, void *pExtra, AFX_CMDHANDLERINFO* pHandler);


BOOL CMainFrame::OnCmdMsg(UINT id, int code, void *pExtra, AFX_CMDHANDLERINFO* pHandler)
{
if (!a->IsVisible()&&m_pTabbedBar->IsVisible())
{
a>DockToWindow(e, CBRS_RIGHT);
a>AttachToTabWnd(e, DM_SHOW, FALSE, &m_pTabbedBar);
}

if (!b->IsVisible() && m_pTabbedBar->IsVisible())
{
b->DockToWindow(a, CBRS_RIGHT);
b->AttachToTabWnd(a, DM_SHOW, FALSE, &m_pTabbedBar);
}

if (!c->IsVisible() && m_pTabbedBar->IsVisible())
{
c->DockToWindow(b, CBRS_RIGHT);
c->AttachToTabWnd(b, DM_SHOW, FALSE, &m_pTabbedBar);
}

if (!d->IsVisible() && m_pTabbedBar->IsVisible())
{
d->DockToWindow(c, CBRS_RIGHT);
d->AttachToTabWnd(c, DM_SHOW, FALSE, &m_pTabbedBar);
}

if (!e->IsVisible() && m_pTabbedBar->IsVisible())
{
e->DockToWindow(d, CBRS_RIGHT);
e->AttachToTabWnd(d, DM_SHOW, FALSE, &m_pTabbedBar);
}

return CFrameWndEx::OnCmdMsg(id, code, pExtra, pHandler);
}

也就是說:這個消息函數會不斷執行,每次執行時,我都判斷一下選項卡是否可見,如果選項卡可見而子面板不可見了,就要將子面板重新顯示出來,五個子面板依次構成一個依附循環,就解決了雙擊和拖拽時消失的問題。希望這個方法能對大家有所幫助。


提出同樣問題的還有:http://bbs.csdn.net/topics/391999680

希望這位朋友已經用其他的方法解決問題了,如果能看到,希望對你有所幫助。


有什麼問題可以給我留言,大家共同進步。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章