科技行者

行者学院 转型私董会 科技行者专题报道 网红大战科技行者

知识库

知识库 安全导航

至顶网软件频道基础软件VC++实现工具栏上添加平面组合框控件

VC++实现工具栏上添加平面组合框控件

  • 扫一扫
    分享文章到微信

  • 扫一扫
    关注官方公众号
    至顶头条

使用过OFFICE2000的人都知道,它的界面,尤其是菜单和工具条,可谓是让人耳目一新。

作者:刘涛 来源:天极开发 2007年10月16日

关键字: 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\n");
 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\n");
   return -1; // fail to create
  }
  if (!m_wndStatusBar.Create(this) || !m_wndStatusBar.SetIndicators(indicators,
sizeof(indicators)/sizeof(UINT)))
  {
   TRACE0("Failed to create status bar\n");
   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\n");
    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已有的开发环境支持,在工具条中加入了平面组合框控件,并实现了组合框的消息响应,用户选择组合框中的某一项后,会弹出一个对话框,提示用户所选择的信息。

查看本文来源

    • 评论
    • 分享微博
    • 分享邮件
    邮件订阅

    如果您非常迫切的想了解IT领域最新产品与技术信息,那么订阅至顶网技术邮件将是您的最佳途径之一。

    重磅专题
    往期文章
    最新文章