程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> VC >> vc教程 >> VC++實現工具欄上添加平面組合框控件

VC++實現工具欄上添加平面組合框控件

編輯:vc教程

  使用過OFFICE2000的人都知道,它的界面,尤其是菜單和工具條,可謂是讓人耳目一新。雖然Visual C++開發工具也提供了對工具條的支持,但通常只是按紐的集合,不能直接加入組合框等控件,實現Office2000風格的工具條。本實例針對其中的一個細節,講述了在Windows環境下用Visual C++6.0在工具條中加入平面組合框控件方法,並實現了組合框的消息響應函數,使得我們的程序看上去更加專業。

  一、實現方法

  用應用程序向導(AppWizard)生成一個基於單文檔的工程(Project),首先打開VC的工具條資源編輯器,在工具條要加入組合框的地方加一個空按紐,並將資源共享ID定義為ID_TOOL_ZOOM。

  其次,從面向對象的思想出發,一個工具條作為一個整體,應該封裝為一個類,組合框控件應該作為這個類的一個成員變量。因此用Visual C++的類向導CLASSWIZARD生成一個以CToolBar為基類的的新類CMainToolBar,並加入成員變量CFlatComboBox m_combobox(CflatComboBox為平面組合框類。

  在向工具條添加控件的過程中,調用CToolBar::GetItemID()函數來獲取每個按鈕的ID,直到搜索到"空"按鈕。CToolBar::GetItemID()函數的原型為:UINT GetItemID( int nIndex ) const,參數nIndex為當前按鈕在工具條中的索引號,該索引號的基准值為"0"。找到"空"按鈕後,調用CToolBar::SetButtonInfo()函數設置按鈕的寬度信息。最後調用CComBox::Create()、CcomBox::AddString()等函數動態創建平面組合框控件,下面的代碼實現了平面組合框控件的動態創建:

//設置指定工具項的寬度並獲取新的區域 80是寬度
m_wndToolBar.SetButtonInfo(index, ID_TOOL_ZOOM, TBBS_SEPARATOR, 80);
m_wndToolBar.GetItemRect(index, &rect);
//設置位置
rect.top+=2;
rect.bottom += 200;
// 創建並顯示
if (!m_wndToolBar.m_wndZoom.Create(WS_CHILD|WS_VISIBLE |
 CBS_AUTOHSCROLL|CBS_DROPDOWNLIST |
 CBS_HASSTRINGS ,rect, &m_wndToolBar, ID_TOOL_ZOOM))
{
 TRACE0("Failed to create combo-box ");
 return FALSE;
}
m_wndToolBar.m_wndZoom.ShowWindow(SW_SHOW);
//填充內容
m_wndToolBar.m_wndZoom.AddString("25%");
m_wndToolBar.m_wndZoom.AddString("50%");
m_wndToolBar.m_wndZoom.AddString("75%");
m_wndToolBar.m_wndZoom.AddString("100%");
m_wndToolBar.m_wndZoom.AddString("125%");
m_wndToolBar.m_wndZoom.AddString("150%");
m_wndToolBar.m_wndZoom.AddString("175%");
m_wndToolBar.m_wndZoom.AddString("200%");
m_wndToolBar.m_wndZoom.SetCurSel(3);

  但是僅僅產生平面組合框是不夠的,必須實現組合框的消息響應函數,才能方便地運用組合框。在Vsiaul C++中,消息響應函數通常都是用類向導來實現,但是此處由於組合框是用函數創建的,所以必須親自動手來寫代碼,也並不麻煩,與類向導生成的代碼格式是一樣的,可以參照來寫。下面代碼定義了組合框的選擇變化消息響應函數:

//////////////////////////////////////////////////////////////////////
BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
//{{AFX_MSG_MAP(CMainFrame)
 ON_WM_CREATE()
 ON_CBN_SELENDOK(ID_TOOL_ZOOM, OnSelectZoomed)
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
//////////////////////////////////////////////////////////////////////
afx_msg void OnSelectZoomed(); 

  二、編程步驟

  1、啟動Visual C++6.0,生成一個單文檔項目,將該項目命名為"ToolBar";

  2、通過資源編輯器新增一個工具按鈕,"Caption"設置為空,ID資源標志符命名為ID_TOOL_ZOOM;

  3、啟動Class Wizard從CToolBar派生一個新類CMainToolBar;

  4、在MainFrm.h文件中添加#include "MainToolBar.h"語句,然後找到 CToolBar m_wndToolBar語句,用CMainToolBar代替CToolBar;

  5、添加代碼,編譯運行程序。

  三、程序代碼

//////////////////////////////////////////////////////////////////////////////
// FlatComboBox.h : header file
#if !defined(FLATCOMBOBOX_H_INCLUDED)
#define FLATCOMBOBOX_H_INCLUDED
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#define FC_DRAWNORMAL 0x00000001
#define FC_DRAWRAISED 0x00000002
#define FC_DRAWPRESSD 0x00000004

// CFlatComboBox window
class CFlatComboBox : public CComboBox
{
 // Construction
 public:
  CFlatComboBox();
  // Attributes
 public:
  bool m_bLBtnDown;
  COLORREF m_clrHilite;
  COLORREF m_clrShadow;
  COLORREF m_clrDkShad;
  COLORREF m_clrButton;
  // Operations
 public:
  void DrawCombo(DWord dwStyle, COLORREF clrTopLeft, COLORREF clrBottomRight);
  int Offset();
  // Overrides
  // ClassWizard generated virtual function overrides
  //{{AFX_VIRTUAL(CFlatComboBox)
  //}}AFX_VIRTUAL
  // Implementation
 public:
  virtual ~CFlatComboBox();
  // Generated message map functions
 protected:
  //{{AFX_MSG(CFlatComboBox)
   afx_msg void OnMouseMove(UINT nFlags, CPoint point);
   afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
   afx_msg void OnLButtonUp(UINT nFlags, CPoint point);
   afx_msg void OnTimer(UINT nIDEvent);
   afx_msg void OnPaint();
  //}}AFX_MSG
  DECLARE_MESSAGE_MAP()
};
#endif // !defined(FLATCOMBOBOX_H_INCLUDED)

///////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "FlatComboBox.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif

///////////////////////////////// CFlatComboBox

CFlatComboBox::CFlatComboBox()
{
 m_bLBtnDown = false;
}

CFlatComboBox::~CFlatComboBox()
{}

BEGIN_MESSAGE_MAP(CFlatComboBox, CComboBox)
 //{{AFX_MSG_MAP(CFlatComboBox)
  ON_WM_MOUSEMOVE()
  ON_WM_LBUTTONDOWN()
  ON_WM_LBUTTONUP()
  ON_WM_TIMER()
  ON_WM_PAINT()
 //}}AFX_MSG_MAP
END_MESSAGE_MAP()

///////////////////////////////////////////////////// CFlatComboBox message handlers
void CFlatComboBox::OnMouseMove(UINT nFlags, CPoint point)
{
 SetTimer(1,10,NULL);
 CComboBox::OnMouseMove(nFlags, point);
}

void CFlatComboBox::OnLButtonDown(UINT nFlags, CPoint point)
{
 m_bLBtnDown = true;
 CComboBox::OnLButtonDown(nFlags, point);
}

void CFlatComboBox::OnLButtonUp(UINT nFlags, CPoint point)
{
 m_bLBtnDown = false;
 Invalidate();
 CComboBox::OnLButtonUp(nFlags, point);
}

void CFlatComboBox::OnTimer(UINT nIDEvent)
{
 POINT pt;
 GetCursorPos(&pt);
 CRect rcItem;
 GetWindowRect(&rcItem);
 static bool bPainted = false;
 // OnLButtonDown, show pressed.
 if (m_bLBtnDown==true) {
  KillTimer (1);
  if (bPainted == true) {
   DrawCombo(FC_DRAWPRESSD, ::GetSysColor(COLOR_BTNSHADOW),::GetSysColor(COLOR_BTNHIGHLIGHT));
   bPainted = false;
  }
  return;
 }
 // If mouse leaves, show flat.
 if (!rcItem.PtInRect(pt)) {
  KillTimer (1);
  if (bPainted == true) {
   DrawCombo(FC_DRAWNORMAL, ::GetSysColor(COLOR_BTNFACE),::GetSysColor(COLOR_BTNFACE));
   bPainted = false;
  }
  return;
 }
 // On mouse over, show raised.
 else {
  if (bPainted == true)
   return;
  else {
   bPainted = true;
   DrawCombo(FC_DRAWRAISED, ::GetSysColor(COLOR_BTNSHADOW),::GetSysColor(COLOR_BTNHIGHLIGHT));
  }
 }
 CComboBox::OnTimer(nIDEvent);
}

void CFlatComboBox::OnPaint()
{
 Default();
 DrawCombo(FC_DRAWNORMAL, ::GetSysColor(COLOR_BTNFACE),::GetSysColor(COLOR_BTNFACE));
}

void CFlatComboBox::DrawCombo(DWord dwStyle, COLORREF clrTopLeft, COLORREF clrBottomRight)
{
 CRect rcItem;
 GetClIEntRect(&rcItem);
 CDC* pDC = GetDC();
 // Cover up dark 3D shadow.
 pDC->Draw3dRect(rcItem, clrTopLeft, clrBottomRight);
 rcItem.DeflateRect(1,1);
 if (!IsWindowEnabled()) {
  pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNHIGHLIGHT),::GetSysColor(COLOR_BTNHIGHLIGHT));
 }
 else {
  pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNFACE),::GetSysColor(COLOR_BTNFACE));
 }
 // Cover up dark 3D shadow on drop arrow.
 rcItem.DeflateRect(1,1);
 rcItem.left = rcItem.right-Offset();
 pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNFACE),::GetSysColor(COLOR_BTNFACE));
 // Cover up normal 3D shadow on drop arrow.
 rcItem.DeflateRect(1,1);
 pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNFACE),::GetSysColor(COLOR_BTNFACE));
 if (!IsWindowEnabled()) {
  return;
 }
 switch (dwStyle)
 {
  case FC_DRAWNORMAL:
   rcItem.top -= 1;
   rcItem.bottom += 1;
   pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNHIGHLIGHT),::GetSysColor(COLOR_BTNHIGHLIGHT));
   rcItem.left -= 1;
   pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNHIGHLIGHT),::GetSysColor(COLOR_BTNHIGHLIGHT));
   break;
  case FC_DRAWRAISED:
   rcItem.top -= 1;
   rcItem.bottom += 1;
   pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNHIGHLIGHT),::GetSysColor(COLOR_BTNSHADOW));
   break;
  case FC_DRAWPRESSD:
   rcItem.top -= 1;
   rcItem.bottom += 1;
   rcItem.OffsetRect(1,1);
   pDC->Draw3dRect(rcItem, ::GetSysColor(COLOR_BTNSHADOW),::GetSysColor(COLOR_BTNHIGHLIGHT));
   break;
 }
 ReleaseDC(pDC);
}

int CFlatComboBox::Offset()
{
 // Thanks to Todd Brannam for this suggestion...
 return ::GetSystemMetrics(SM_CXHTHUMB);
}

/////////////////// MainToolBar.h: interface for the CMainToolBar class.

#if !defined(AFX_MAINTOOLBAR_H__76CF28F4_005F_11D7_8F58_00E04C0BECE6__INCLUDED_)
#define AFX_MAINTOOLBAR_H__76CF28F4_005F_11D7_8F58_00E04C0BECE6__INCLUDED_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "FlatComboBox.h"

class CMainToolBar : public CToolBar
{
 public:
  CMainToolBar();
  virtual ~CMainToolBar();
 public:
  CFlatComboBox m_wndZoom;
};
#endif

////////////////////////////// MainToolBar.cpp: implementation of the CMainToolBar class.
#include "stdafx.h"
#include "ToolBar.h"
#include "MainToolBar.h"
#ifdef _DEBUG
#undef THIS_FILE
static char THIS_FILE[]=__FILE__;
#define new DEBUG_NEW
#endif

CMainToolBar::CMainToolBar()
{}

CMainToolBar::~CMainToolBar()
{}

//////////////////////////////////////////////////////////////////////////////////////////////
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
 if (CFrameWnd::OnCreate(lpCreateStruct) == -1)
  return -1;
 if (!m_wndToolBar.CreateEx(this, TBSTYLE_FLAT, WS_CHILD | WS_VISIBLE |
   CBRS_TOP| CBRS_GRIPPER | CBRS_TOOLTIPS | CBRS_FLYBY |
   CBRS_SIZE_DYNAMIC) ||!m_wndToolBar.LoadToolBar(IDR_MAINFRAME))
 {
  TRACE0("Failed to create toolbar ");
  return -1; // fail to create
 }
 if (!m_wndStatusBar.Create(this) || !m_wndStatusBar.SetIndicators(indicators,
sizeof(indicators)/sizeof(UINT)))
 {
  TRACE0("Failed to create status bar ");
  return -1; // fail to create
 }
 // TODO: Delete these three lines if you don’t want the toolbar to be dockable
 m_wndToolBar.EnableDocking(CBRS_ALIGN_ANY);
 EnableDocking(CBRS_ALIGN_ANY);
 DockControlBar(&m_wndToolBar);
 int index = 0;
 RECT rect;
 //找到指定的工具項
 while(m_wndToolBar.GetItemID(index)!=ID_TOOL_ZOOM)index++;
 //設置指定工具項的寬度並獲取新的區域 80是寬度
 m_wndToolBar.SetButtonInfo(index, ID_TOOL_ZOOM, TBBS_SEPARATOR, 80);
 m_wndToolBar.GetItemRect(index, &rect);
 //設置位置
 rect.top+=2;
 rect.bottom += 200;
 // 創建並顯示
 if (!m_wndToolBar.m_wndZoom.Create(WS_CHILD|WS_VISIBLE |CBS_AUTOHSCROLL|CBS_DROPDOWNLIST |
CBS_HASSTRINGS ,rect, &m_wndToolBar, ID_TOOL_ZOOM))
 {
  TRACE0("Failed to create combo-box ");
  return FALSE;
 }
 m_wndToolBar.m_wndZoom.ShowWindow(SW_SHOW);
 //填充內容
 m_wndToolBar.m_wndZoom.AddString("25%");
 m_wndToolBar.m_wndZoom.AddString("50%");
 m_wndToolBar.m_wndZoom.AddString("75%");
 m_wndToolBar.m_wndZoom.AddString("100%");
 m_wndToolBar.m_wndZoom.AddString("125%");
 m_wndToolBar.m_wndZoom.AddString("150%");
 m_wndToolBar.m_wndZoom.AddString("175%");
 m_wndToolBar.m_wndZoom.AddString("200%");
 m_wndToolBar.m_wndZoom.SetCurSel(3);
 return 0;
}

void CMainFrame::OnSelectZoomed()
{
 CString strContent;
 m_wndToolBar.m_wndZoom.GetWindowText(strContent);
 AfxMessageBox(strContent);
}

  四、小結

  為了實現Office2000風格的工具條,本實例介了一種比較巧妙的方法,利用Visual C++6.0已有的開發環境支持,在工具條中加入了平面組合框控件,並實現了組合框的消息響應,用戶選擇組合框中的某一項後,會彈出一個對話框,提示用戶所選擇的信息。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved