Chinaunix首页 | 论坛 | 博客
  • 博客访问: 589222
  • 博文数量: 752
  • 博客积分: 40000
  • 博客等级: 大将
  • 技术积分: 5005
  • 用 户 组: 普通用户
  • 注册时间: 2008-10-13 14:47
文章分类

全部博文(752)

文章存档

2011年(1)

2008年(751)

我的朋友

分类:

2008-10-13 16:48:41

设计XP风格的按钮
作者:



论坛上许多人都在讨论如何编写具有XP风格的界面,其实网上有设计好的类库,可以直接拿来使用。但这些终归是别人写的,能不能转化成自已的呢。于是笔者就对这些代码进行研究,算是偷一点吧:)
研究了几种控件,这里就把其中最简单的按钮控件拿来供大家一起讨论。
这是程序的运行效果:

步骤:
1、创建一个派生自CButton的新类CButtonXp
2、重载PreSubClassWindow()函数,在该函数内修改按钮的风格为自绘制(owner):
添加如下代码:ModifyStyle(0,BS_OWNERDRAW);
3、因为XP风格按钮具有鼠标感应的效果,当鼠标移动到按钮上方时,按钮的颜色会改变。所以就必须跟踪鼠标。当鼠标移到按钮上方时,窗口会收到
WM_MOUSEMOVE消息,但怎么才能得知鼠标离开按钮呢?
这里我们使用 TrackMouseEvent() Api函数:
BOOL TrackMouseEvent(  LPTRACKMOUSEEVENT lpEventTrack  );
参数:
typedef struct tagTRACKMOUSEEVENT {
DWORD cbSize; //结构大小
DWORD dwFlags; //设定为TME_LEAVE
HWND hwndTrack; //要跟踪鼠标的窗口句柄
DWORD dwHoverTime;} TRACKMOUSEEVENT, *LPTRACKMOUSEEVENT;
调用该函数可以在鼠标离开指定窗口时收到WM_MOUSELEAVE消息。

添加成员变量:m_bOver ,初始化为FALSE。m_bOver=true用来表示鼠标在按钮区域。
添加WM_MOUSEMOVE消息处理函数:
void CButtonXp::OnMouseMove(UINT nFlags, CPoint point) 
{
	if(m_bOver ==FALSE)
	{
		//鼠标在按钮之上
		m_bOver =TRUE;

		//按钮重绘
		InvalidateRect(NULL,FALSE);

		//跟踪鼠标
		//当鼠标离开按钮区域会收到WM_MOUSELEAVE,该消息直接调用OnMouseOut()
		TRACKMOUSEEVENT	tme;
		tme.cbSize =sizeof(TRACKMOUSEEVENT);
		tme.dwFlags =TME_LEAVE;
		tme.dwHoverTime=0;
		tme.hwndTrack =m_hWnd;
		::TrackMouseEvent(&tme);
	}
	CButton::OnMouseMove(nFlags, point);
}
再添加一成员函数OnMouseOut(),
并在BEGIN_MESSAGE_MAP(CButtonXp, CButton)和END_MESSAGE_MAP()之间添加
宏 ON_MESSAGE(WM_MOUSELEAVE,OnMouseOut)

在OnMouseOut()中写入以下代码

void CButtonXp::OnMouseOut ()
{
	//鼠标已离开按钮区域
    m_bOver =FALSE;

	//重绘按钮
	InvalidateRect(NULL,FALSE);
	
}
4、添加成员函数 MouseOver()
//返回鼠标是否在按钮区域内
BOOL CButtonXp::MouseOver()
{
	return m_bOver;
}
5、最后重载DrawItem(LPDRAWITEMSTRUCT lpDIS)
void CButtonXp::DrawItem(LPDRAWITEMSTRUCT lpDIS) 
{
	CDC	*pDC =CDC::FromHandle(lpDIS->hDC);

	CRect	rtControl(lpDIS->rcItem);

	CPen	pen,*old_pen;
	CBrush	brush,*old_brush;
	CString  strText;
	
	HFONT hOldFont = (HFONT)pDC->SelectObject ((HFONT)::GetStockObject (DEFAULT_GUI_FONT));

	UINT	state =lpDIS->itemState;

	if(state & ODS_FOCUS)
	{
		rtControl.DeflateRect(1,1);	//拥有焦点矩形变小
	}

	if((state & ODS_DISABLED) ||
		(!MouseOver() &&!(state & ODS_SELECTED)))
	{
		//普通状态、禁用、拥有焦点三种情况下
		pen.CreatePen (PS_SOLID, 1, ::GetSysColor(COLOR_3DSHADOW));
		brush.CreateSolidBrush(HLS_TRANSFORM(::GetSysColor(COLOR_3DFACE),-10,0));
	}
	else
	{
		COLORREF	crBorder =::GetSysColor(COLOR_HIGHLIGHT);

		pen.CreatePen(PS_SOLID, 1, crBorder);

		if( state & ODS_SELECTED)
		{
			//按钮按下时
			brush.CreateSolidBrush(HLS_TRANSFORM(crBorder,+50,-50));
			
			pDC->SetTextColor(RGB(240,240,240));
		}
		else
		{
			//鼠标在区域内	
			brush.CreateSolidBrush(HLS_TRANSFORM(crBorder,+80,-66));
			pDC->SetTextColor(::GetSysColor(COLOR_BTNTEXT));
		}
	}

	if(state &ODS_DISABLED)
		pDC->SetTextColor(::GetSysColor(COLOR_GRAYTEXT));//灰色字:禁用状态
	else if(state & ODS_SELECTED)
		pDC->SetTextColor(RGB(240,240,240));	//白色字:PUSH状态
	else if(MouseOver())	
		pDC->SetTextColor(0);	//黑色字:热感应状态
	else
		pDC->SetTextColor(::GetSysColor(COLOR_BTNTEXT));	//黑色字:普通状态	

	old_brush=pDC->SelectObject(&brush);
	old_pen  =pDC->SelectObject(&pen);

	pDC->Rectangle(rtControl);
	pDC->SetBkMode(TRANSPARENT);
	GetWindowText(strText);
	pDC->DrawText(strText,rtControl,DT_SINGLELINE|DT_CENTER|DT_VCENTER);


	if(state & ODS_FOCUS)
	{
		rtControl.DeflateRect(3,3);
		pDC->DrawFocusRect(rtControl);
	}

	pDC->SelectObject(old_pen);
	pDC->SelectObject(old_brush);
	pDC->SelectObject(hOldFont);
}
还有一个要注意的是,要使用TrackMouseEvent(),必须在加入头文件winuser.h 和extern "C" WINUSERAPI BOOL WINAPI TrackMouseEvent(LPTRACKMOUSEEVENT lpEventTrack);

本程序在win98 + Visual C++ 6.0环境下编译通过.


--------------------next---------------------

看不太懂了。 ( ashengsheng 发表于 2006-6-1 20:19:00)
 
XP下我用VC++.NET 7.1(2003)编译有错误,不知道为什么...
g:\cpp\Myqq\Myqq\ButtonXP.cpp(26) : error C2440: “static_cast” : 无法从“LPARAM (__thiscall CButtonXP::* )(void)”转换为“LRESULT (__thiscall CWnd::* )(WPARAM,LPARAM)”
        在匹配目标类型的范围内没有具有该名称的函数 ( bjtuwei 发表于 2006-3-1 17:24:00)
 
( tfq2002 发表于 2005-9-21 16:41:00)
 
图片怎么发不上去 ( tfq2002 发表于 2005-9-21 16:39:00)
 
程序有个bug,如果把
CButtonXp::DrawItem(LPDRAWITEMSTRUCT lpDIS) 
{
······
if(state & ODS_FOCUS)
{
  rtControl.DeflateRect(1,1);//拥有焦点矩形变小
}

······
}
改为rtControl.DeflateRect(5,5);
会出现:
[img]d:\1.gif[/img] ( tfq2002 发表于 2005-9-21 16:37:00)
 
不错,非常好,谢谢! ( ikohl 发表于 2005-6-16 14:40:00)
 
'HLS_TRANSFORM' : undeclared identifier ??? ( umama 发表于 2005-5-11 10:46:00)
 
不错,谢谢,以后可以实现自己的xp风格了,哈哈 ( zbc 发表于 2003-8-4 13:08:00)
 
XP下我用VC++.NET 7.1(2003)编译有错误,不知道为什么... ( namelysweet 发表于 2003-7-12 15:21:00)
 
如果用SetCapture()捕获鼠标消息的话,那TrackMouseEvent()还有什么用列???? ( kinglz 发表于 2003-3-19 8:50:00)
 
.......................................................

--------------------next---------------------

阅读(229) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~