引言
做用户界面的时候经常要用到一些静态文本控件,显示一些文字信息,但是 MFC 提供的 CStatic类的功能过于简单,无法满足高级需求。为此我从 CStatic 派生了一个类 CLabelEx,扩展了CStatic。
一、功能简介
新增的功能主要有:
1、设置背景图片SetBGBitmap();设置鼠标经过时的背景图片SetMouseOverBGBitmap();设置鼠标单击后的背景图片SetClickedBGBitmap();
2、设置标签图片,SetLabelBitmap();设置鼠标经过时的标签图片SetMouseOverLabelBitmap();设置鼠标单击时的标签图片
3、文字功能
(1)设置字体颜色,下划线等就不说了. (2)感应鼠标经过时自动加上下划线,自动把文字变蓝(就像一个超链接一样)
4、边框和背景
可以设置/取消边框,指定边框颜色;设置背景色,并填充整个标签
二、实现原理
1、其实就是在OnPaint()里画出各种效果:
void CLabelEx::OnPaint() { CPaintDC dc(this); // device context for painting dc.SetTextColor(m_crText); dc.SetBkMode(TRANSPARENT); dc.SelectObject(this->GetFont()); ///准备工作 CRect rect; CDC MemDC; CPen BorderPen,*pOldPen,UnderLinePen; CBrush BGBrush,*pOldBrush; BITMAP bm; int nTextLeft=0,nTextTop=0; //文字输出的位置
this->GetClientRect(&rect);
MemDC.CreateCompatibleDC(&dc); MemDC.SetMapMode(dc.GetMapMode());
///画边框 if(m_bBorder) { BorderPen.CreatePen(PS_SOLID,1,m_crBorder); BGBrush.CreateSolidBrush(m_crBG);
pOldPen=dc.SelectObject(&BorderPen); pOldBrush=dc.SelectObject(&BGBrush);
dc.Rectangle(&rect);
dc.SelectObject(pOldPen); dc.SelectObject(pOldBrush);
rect.DeflateRect(1,1); } ///贴背景图 if(m_bClicked && m_ClickedBGBm.GetSafeHandle()!=NULL) { MemDC.SelectObject(m_ClickedBGBm); dc.BitBlt(rect.left,rect.top,rect.Width(),rect.Height(), &MemDC,0,0,SRCCOPY); } else if(m_bOver && m_MouseOverBGBm.GetSafeHandle()!=NULL)//鼠标经过的时候 { MemDC.SelectObject(m_MouseOverBGBm); dc.BitBlt(rect.left,rect.top,rect.Width(),rect.Height(), &MemDC,0,0,SRCCOPY); } else if(m_BGBm.GetSafeHandle()!=NULL) { MemDC.SelectObject(m_BGBm); dc.BitBlt(rect.left,rect.top,rect.Width(),rect.Height(), &MemDC,0,0,SRCCOPY); } ///贴标签图片 if(m_bClicked && m_ClickedLabelBm.GetSafeHandle()!=NULL) { m_ClickedLabelBm.GetBitmap(&bm); double fScal=bm.bmWidth*1.0/bm.bmHeight; nTextLeft=int(rect.Height()*fScal)+4; MemDC.SelectObject(m_ClickedLabelBm); dc.StretchBlt(rect.left,rect.top,int(rect.Height()*fScal),rect.Height(), &MemDC,0,0,bm.bmWidth,bm.bmHeight,SRCCOPY); } else if(m_bOver && m_MouseOverLabelBm.GetSafeHandle()!=NULL) { m_MouseOverLabelBm.GetBitmap(&bm); double fScal=bm.bmWidth*1.0/bm.bmHeight; nTextLeft=int(rect.Height()*fScal)+4; MemDC.SelectObject(m_MouseOverLabelBm); dc.StretchBlt(rect.left,rect.top,int(rect.Height()*fScal),rect.Height(), &MemDC,0,0,bm.bmWidth,bm.bmHeight,SRCCOPY); } else if(m_LabelBm.GetSafeHandle()!=NULL) { m_LabelBm.GetBitmap(&bm); double fScal=bm.bmWidth*1.0/bm.bmHeight; nTextLeft=int(rect.Height()*fScal)+4; MemDC.SelectObject(m_LabelBm); dc.StretchBlt(rect.left,rect.top,int(rect.Height()*fScal),rect.Height(), &MemDC,0,0,bm.bmWidth,bm.bmHeight,SRCCOPY); } else { nTextLeft=4; } ///输出文字 TEXTMETRIC tm; dc.GetTextMetrics(&tm); CString strText; this->GetWindowText(strText); nTextTop=rect.top+(rect.Height()-tm.tmHeight)/2; if(strText.GetLength()>0) { dc.TextOut(nTextLeft,nTextTop,strText); }
///画下划线 if(m_bUnderLine) { nTextLeft-=2; nTextTop=nTextTop+tm.tmHeight+1; UnderLinePen.CreatePen(PS_SOLID,1,m_crUnderLine); pOldPen=dc.SelectObject(&UnderLinePen); dc.MoveTo(nTextLeft,nTextTop); dc.LineTo(nTextLeft+tm.tmAveCharWidth*strText.GetLength(),nTextTop); } } |
注:对字体加下划线我没有使用直接设置字体下划线的方法,因为我觉得那样不好看,呵呵
|