本文將介紹兩個可復用的C++類:CXTOutlookBar 和 CXTPagerCtrl,用它們可以實現Outlook風格的用戶界面,這兩個類出自Codejock軟件公司,是其產品Xtreme Toolkit的一部分。根據該公司的許可條款,任何人都可以不受限制地免費使用這兩個類的源代碼。
介紹 CXTOutlookBar類派生于ListBox,主要實現 Outlook 界面式樣控制。CXTPagerCtrl類用于容納和滾動CXTOutlookBar窗口,這個類包裝了與Windows窗口管理有關的(CWnd)API。兩個類的使用都很簡單,用法與標準的MFC類庫一樣,沒有什么特別要求。
實現 本文附帶的例子程序是一個標準的MFC SDI應用程序(如圖一所示),

圖一 例子程序運行畫面
這個程序的主框架是兩個切分的視圖:左邊的視圖為 COutbarView,它派生于CView,包含Outlook式樣控制機制和窗口管理;右邊的視圖為應用程序向導生成的視類,你在應用程序向導中可以自己規定這個類從哪個基類派生,例子程序是從CListView派生的。如果想在自己的程序中使用Outlook式樣的控制,只要在包含Outlook式樣的視類(如本文例子程序的COutbarView)中聲明實例即可,例如:
// 屬性
protected:
CXTOutlookBar m_wndOutlookBar;
CXTPagerCtrl m_wndPager;
然后在視類的WM_CREATE/OnCreate消息處理例程中創建窗口控制和Outlook菜單。此時還要添加Outlook菜單項并對它們進行初始化,并設置好按鈕的尺寸,創建子窗口:
int COutbarView::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CView::OnCreate(lpCreateStruct) == -1)
return -1;
// 創建頁窗口
if (!m_wndPager.Create(WS_CHILD|WS_VISIBLE|PGS_VERT,
CRect(0,0,0,0), this, IDC_PAGER_CTRL ))
{
TRACE0("Failed to create CPagerCtrl...\n");
return -1;
}
// 以m_wndPager作為父窗口創建 Outlook 式樣控制
if (!m_wndOutlookBar.Create( WS_CHILD | WS_VISIBLE | WS_TABSTOP,
CRect(0,0,0,0), &m_wndPager, IDC_OUTBAR ))
{
TRACE0("Failed to create COutlookBar...\n");
return -1;
}
// 設置接受消息的 CWnd 對象
m_wndOutlookBar.SetOwner(this);
m_wndOutlookBar.SetColors(RGB(0xff,0xff,0xff), RGB(0x3a,0x6e,0xa5));
// 添加 Outlook 控制菜單項
m_wndOutlookBar.AddMenuItem(IDI_ICON_LOGO, _T("我的VC知識庫") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_NOTES, _T("技術論壇") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_JOURNAL, _T("在線雜志") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_HLIGHT, _T("精華區") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_SOURCE, _T("源代碼") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_COOLLIB, _T("酷庫") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_VCKBASE, _T("VCKBASE Today")),
m_wndOutlookBar.AddMenuItem(IDI_ICON_PUBLIC, _T("開發聯盟") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_CONTACTS, _T("VC知識庫") ),
m_wndOutlookBar.AddMenuItem(IDI_ICON_DELETED, _T("垃圾箱")),
// 在指定的索引處插入菜單項
m_wndOutlookBar.InsertMenuItem(0, IDI_ICON_INBOX, _T("收件箱") ),
m_wndOutlookBar.InsertMenuItem(1, IDI_ICON_OUTBOX, _T("發件箱") ),
m_wndOutlookBar.InsertMenuItem(2, IDI_ICON_CALENDAR, _T("日歷") ),
// 設置 COutlookBar 所管理的子窗口以及按鈕的尺寸(=15)
m_wndPager.SetChild(m_wndOutlookBar.GetSafeHwnd());
m_wndPager.SetButtonSize(15);
return 0;
}
接下來我們對窗口的大小進行管理,因此要編寫WM_SIZE/OnSize代碼,這樣將保證視圖大小改變后頁控制會作相應的位置調整,這一步你可以用類向導來做。
void COutbarView::OnSize(UINT nType, int cx, int cy)
{
CView::OnSize(nType, cx, cy);
if(m_wndPager.GetSafeHwnd()) {
m_wndPager.MoveWindow(0,0,cx,cy);
}
}
最后,我們的任務是添加針對頁控制的PGN_SCROLL和PGN_CALCSIZE消息處理代碼。它告訴我們何時有頁滾動并允許設置Outlook菜單窗口的可滾動尺寸。除了PGN_消息處理之外,我們還需要添加XTWM_OUTBAR_NOTIFY消息處理。它將通知我們何時用戶點擊了Outlook菜單項。為此在COutbarView類的實現文件中(.cpp)添加下面的消息映射:
BEGIN_MESSAGE_MAP(COutbarView, CView)
//{{AFX_MSG_MAP(COutbarView)
...
//}}AFX_MSG_MAP
ON_MESSAGE(XTWM_OUTBAR_NOTIFY, OnOutbarNotify)
ON_NOTIFY(PGN_SCROLL, IDC_PAGER_CTRL, OnPagerScroll)
ON_NOTIFY(PGN_CALCSIZE, IDC_PAGER_CTRL, OnPagerCalcSize)
END_MESSAGE_MAP()
同時在實現文件中添加下面的成員函數:
BOOL COutbarView::OnPagerCalcSize(NMPGCALCSIZE* pNMPGCalcSize, LRESULT* pResult)
{
switch(pNMPGCalcSize->dwFlag)
{
case PGF_CALCWIDTH:
break;
case PGF_CALCHEIGHT:
pNMPGCalcSize->iHeight = m_wndOutlookBar.GetCount()
*(::GetSystemMetrics(SM_CYICON)*2);
break;
}
*pResult = 0;
return TRUE;
}
BOOL COutbarView::OnPagerScroll(NMPGSCROLL* /*pNMPGScroll*/, LRESULT* pResult)
{
*pResult = 0;
return TRUE;
}
void COutbarView::OnOutbarNotify(UINT lParam, LONG wParam)
{
switch( wParam ) // 控制 id.
{
case IDC_OUTBAR:
{
// 獲得菜單項
XT_CONTENT_ITEM* pContentItems =
m_wndOutlookBar.GetMenuItem((int)lParam);
ASSERT(pContentItems);
AfxMessageBox(pContentItems->m_strText);
}
break;
}
}
在頭文件中添加
// 產生消息映射函數
protected:
//{{AFX_MSG(COutbarView)
...
//}}AFX_MSG
afx_msg BOOL OnPagerScroll(NMPGSCROLL* pNMPGScroll, LRESULT * pResult);
afx_msg BOOL OnPagerCalcSize(NMPGCALCSIZE * pNMPGCalcSize, LRESULT* pResult);
DECLARE_MESSAGE_MAP()
大功告成,編譯運行程序吧......
雖然這個例子只是一個Demo,但你如果有興趣,可以很容易擴充它的功能,使它更實用一些。