VC中给树形控件的图标加上工具提示

发表于:2007-07-01来源:作者:点击数: 标签:
我从没有在任何一个应用程序中看到过图标的工具提示。有时候查遍了整个帮助文档也没有明白某个图标是什么意思。如果能在自己的程序中为图标加上工具提示,一定会使界面的友好性大大增加。本文中以树形控件为例,详细介绍了在VC中使用MFC提供的机制来实现图标


我从没有在任何一个应用程序中看到过图标的工具提示。有时候查遍了整个帮助文档也没有明白某个图标是什么意思。如果能在自己的程序中为图标加上工具提示,一定会使界面的友好性大大增加。本文中以树形控件为例,详细介绍了在VC中使用MFC提供的机制来实现图标工具提示的方法。 

第一步:使控件可以显示工具提示 

调用EnableToolTips(TRUE)使一个窗口可以显示工具提示。在什么地方插入这条代码最好呢?在类的PreSubclassWindow()中。因为不管一个控件如何被创建,MFC都会调用此函数。而其他的函数则不一定会被调用。以OnCreate()为例,如果调用Create()或CreateEx()创建一个控件,OnCreate()会被调用,而如果一个控件是从对话框资源创建,OnCreate()就不会被调用。 

实现代码如下:
void CTreeCtrlX::PreSubclassWindow() 
{
CTreeCtrl::PreSubclassWindow();

EnableToolTips(TRUE);
}

第二步:重载虚函数OnToolHitTest() 

MFC调用函数来确定在某个点是否应该显示工具提示。MSDN建议如果鼠标落在应该显示工具提示的点上,返回值1。这并不完全正确。这个函数应该返回不同的值来区分窗口中不同的应该显示提示的区域。 

在这个函数中,本文只处理鼠标落在节点图标或节点状态图标上的情况。读者可以按照自己的情况向树的其他元素上添加工具提示。在两种情况下,都要计算图标的区域,并且把TOOLINFO的uID设为鼠标所在点的树节点的句柄。注意,尽管对于节点图标和节点状态图标,使用了相同的id,但返回值并不相同。不同的返回值迫使MFC更新工具提示。 

虽然我们可以在此函数中给出工具提示,但因为鼠标的每次移动都会调用此函数,太多的处理并不是一个好注意,所以我们在其他的函数中处理应该显示什么提示的问题。 

类声明中的代码如下所示:
// Overrides
// ClassWizard generated 
virtual function overrides
//{{AFX_VIRTUAL(CTreeCtrlX)
protected:

virtual int OnToolHitTest
( CPoint point, TOOLINFO* pTI ) const;
//}}AFX_VIRTUAL

实现代码如下所示:
int CTreeCtrlX::OnToolHitTest
(CPoint point, TOOLINFO * pTI) const
{
RECT rect;

UINT nFlags;
HTREEITEM hitem = HitTest( point, &nFlags );
if( nFlags & TVHT_ONITEMICON )
{
CImageList *pImg = GetImageList( TVSIL_NORMAL );
IMAGEINFO imageinfo;
pImg- >GetImageInfo( 0, &imageinfo );

GetItemRect( hitem, &rect, TRUE );
rect.right = rect.left - 2;
rect.left -= (imageinfo.rcImage.right + 2);

pTI- >hwnd = m_hWnd;
pTI- >uId = (UINT)hitem;
pTI- >lpszText = LPSTR_TEXTCALLBACK;
pTI- >rect = rect;
return pTI- >uId;
}
else if( nFlags & TVHT_ONITEMSTATEICON )
{
CImageList *pImg = GetImageList( TVSIL_NORMAL );
IMAGEINFO imageinfo;
pImg- >GetImageInfo( 0, &imageinfo );

GetItemRect( hitem, &rect, TRUE );
rect.right = rect.left - 
(imageinfo.rcImage.right + 2);

pImg = GetImageList( TVSIL_STATE );
rect.left = rect.right - imageinfo.rcImage.right ;

pTI- >hwnd = m_hWnd;
pTI- >uId = (UINT)hitem;
pTI- >lpszText = LPSTR_TEXTCALLBACK;
pTI- >rect = rect;

// 返回与节点图标不同的值
return pTI- >uId*2; 
}
return -1;
}

第三步:处理TTN_NEEDTEXT消息; 

加入一个函数处理TTN_NEEDTEXT消息通知。当工具处理控制需要知道应该显示什么信息时,这条消息被发出。由于上一步中我们给TOOLINFO的lpszText赋值为LPSTR_TEXTCALLBACK,所以我们要处理这个消息VC的ClassWizard并不支持这条消息被映射,所以只有我们自己加入这条消息的映射机制加入到MESSAGE_MAP中去。我们不得不处理这个消息的两个版本,TTN_NEEDTEXTA和TTN_NEEDTEXTA。消息映射的代码如下所示: 

BEGIN_MESSAGE_MAP(CTreeCtrlX, CTreeCtrl)
//{{AFX_MSG_MAP(CTreeCtrlX)

//}}AFX_MSG_MAP
ON_NOTIFY_EX_RANGE
(TTN_NEEDTEXTW, 0, 0xFFFF, OnToolTipText)
ON_NOTIFY_EX_RANGE
(TTN_NEEDTEXTA, 0, 0xFFFF, OnToolTipText)
END_MESSAGE_MAP()

下面的代码是加到类声明中: 
protected:
//{{AFX_MSG(CTreeCtrlX)

//}}AFX_MSG
afx_msg BOOL OnToolTipText
( UINT id, NMHDR * pNMHDR, LRESULT * pResult );
DECLARE_MESSAGE_MAP()

现在讨论这个函数本身的实现。为了适应不同的语言字符集,ANSI字符集和UNICODE字符集都必须被处理,处理过程会有些不同。此处对树形控件的本身产生的ToolTip消息不予处理,过滤掉上述消息的原则是树形控件本身产生的消息的ID是树形控件窗口的句柄,并且有TTF_IDISHWND标志。根据鼠标位置可以确定应该给出节点图标还是状态图标的工具提示。本文根据笔者画的图显示了一些无关紧要的提示,读者做这一步时应该加入一些有意义的提示。当然,本文假定控件包含节点图标和状态图标。如不包含,计算鼠标位置时要注意 不要计算错误。 

BOOL CTreeCtrlX::OnToolTipText
( UINT id, NMHDR * pNMHDR, LRESULT * pResult )
{
// 需要处理ANSI和UNICODE两种格式
TOOLTIPTEXTA* pTTTA = (TOOLTIPTEXTA*)pNMHDR;
TOOLTIPTEXTW* pTTTW = (TOOLTIPTEXTW*)pNMHDR;
CString strTipText;
UINT nID = pNMHDR- >idFrom;

// 不必处理树自己发出的ToolTip消息
if( nID == (UINT)m_hWnd &&(( pNMHDR- >code == TTN_NEEDTEXTA && 
pTTTA- >uFlags & TTF_IDISHWND ) ||
( pNMHDR- >code == TTN_NEEDTEXTW && 
pTTTW- >uFlags & TTF_IDISHWND ) ) )
return FALSE;

// 得到鼠标位置
const MSG* pMessage;
CPoint pt;
pMessage = GetCurrentMessage();
ASSERT ( pMessage );
pt = pMessage- >pt;
ScreenToClient( &pt );

UINT nFlags;
HTREEITEM hitem = HitTest( pt, &nFlags );
if( nFlags & TVHT_ONITEMICON )
{
int nImage, nSelImage;
GetItemImage( (HTREEITEM ) nID, nImage, nSelImage );
switch(nImage)
{
case 0:
strTipText = "叉";
break;
case 1:
strTipText = "加号";
break;
case 2:
strTipText = "菱形";
break;
}
}
else
{
if( (GetItemState( (HTREEITEM ) nID, 
TVIS_STATEIMAGEMASK ) > >12 ) == 2 )
strTipText.Format( "此节点被选中" );
else
strTipText.Format( "此节点未被选中" );
}

#ifndef _UNICODE
if (pNMHDR- >code == TTN_NEEDTEXTA)
lstrcpyn(pTTTA- >szText, strTipText, 80);
else
_mbstowcsz(pTTTW- >szText, strTipText, 80);
#else
if (pNMHDR- >code == TTN_NEEDTEXTA)
_wcstombsz(pTTTA- >szText, strTipText, 80);
else
lstrcpyn(pTTTW- >szText, strTipText, 80);
#endif
*pResult = 0;

return TRUE; // 消息处理完毕
}

本文程序在Win9x,VC6.0下调试通过。

原文转自:http://www.ltesting.net