MFC库里没有符合这个条件的控件,于是我自己写了一个,初步测试有效。

注:可以设置透明背景,但还不能做到透明度设置(如50%透明度)

如果设置了背景色,就不保留透明背景

默认背景色是透明的

  1. // 设置背景色(若clr为CLR_NONE,则背景透明)
  2. void SetBackgroundColor(COLORREF clr){m_clrBackground = clr;}
  3. // 设置文字前景色
  4. void SetTextColor(COLORREF clr){m_clrText = clr;}
  5. // 设置文字字体
  6. void SetFont(CString strFaceName, LONG nHeight);

如何使用:

1.先将RichStatic.h和RichStatic.cpp添加入工程
    2.对话框添加Static控件后,增加一个控件变量,类型设置为CRichStatic(或手动添加,在对话框类DoDataExchange中添加DDX_Control)

源码:

  1. #pragma once
  2. // CRichStatic
  3. class CRichStatic : public CStatic
  4. {
  5. DECLARE_DYNAMIC(CRichStatic)
  6. public:
  7. CRichStatic();
  8. virtual ~CRichStatic();
  9. protected:
  10. afx_msg BOOL OnEraseBkgnd(CDC* pDC);
  11. afx_msg LRESULT OnSetText(WPARAM,LPARAM);
  12. DECLARE_MESSAGE_MAP()
  13. virtual void PreSubclassWindow();
  14. private:
  15. COLORREF m_clrText;          // 文字前景色
  16. COLORREF m_clrBackground;    // 文字背景色
  17. CFont *m_pTextFont;          // 文字字体
  18. CBitmap m_Bmp;               // 保存背景用的位图对象
  19. public:
  20. // 设置背景色(若clr为CLR_NONE,则背景透明)
  21. void SetBackgroundColor(COLORREF clr){m_clrBackground = clr;}
  22. // 设置文字前景色
  23. void SetTextColor(COLORREF clr){m_clrText = clr;}
  24. // 设置文字字体
  25. void SetFont(CString strFaceName, LONG nHeight);
  26. public:
  27. virtual void DrawItem(LPDRAWITEMSTRUCT /*lpDrawItemStruct*/);
  28. };

  1. // RichStatic.cpp : 实现文件
  2. //
  3. #include "stdafx.h"
  4. #include "RichStatic.h"
  5. // CRichStatic
  6. IMPLEMENT_DYNAMIC(CRichStatic, CStatic)
  7. CRichStatic::CRichStatic():
  8. m_clrText(0), m_clrBackground(CLR_NONE), m_hFont(NULL), m_selfCreated(FALSE),
  9. m_xAlignment(X_LEFT), m_yAlignment(Y_TOP)
  10. {
  11. }
  12. CRichStatic::~CRichStatic()
  13. {
  14. if (m_selfCreated && m_hFont != NULL)
  15. {
  16. DeleteObject(m_hFont);    // 若字体对象为对象自己创建并且不为NULL,则销毁掉以释放内核对象
  17. }
  18. }
  19. BEGIN_MESSAGE_MAP(CRichStatic, CStatic)
  20. ON_MESSAGE(WM_SETTEXT,OnSetText)
  21. ON_WM_ERASEBKGND()
  22. END_MESSAGE_MAP()
  23. // CRichStatic 消息处理程序
  24. void CRichStatic::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
  25. {
  26. if (m_clrBackground != CLR_NONE)    // 若背景色不为CLR_NONE(CLR_NONE表示无背景色),则绘制背景
  27. {
  28. RECT rect;
  29. GetWindowRect(&rect);
  30. CBrush brush;
  31. brush.CreateSolidBrush(m_clrBackground);
  32. ::SelectObject(lpDrawItemStruct->hDC, brush.m_hObject);    // 设置画刷颜色
  33. ::SelectObject(lpDrawItemStruct->hDC, GetStockObject(NULL_PEN));    // 设置笔为空笔(不绘制边界)
  34. Rectangle(lpDrawItemStruct->hDC, 0, 0,rect.right - rect.left, rect.bottom - rect.top);
  35. }
  36. CString strCaption;    // 标题文字
  37. GetWindowText(strCaption);
  38. if (m_hFont != NULL)
  39. {
  40. ::SelectObject(lpDrawItemStruct->hDC, m_hFont);
  41. }
  42. // 计算输出字串的横纵坐标
  43. int x = 0, y = 0;
  44. if (X_LEFT != m_xAlignment || Y_TOP != m_yAlignment)    // 不是左对齐或不是顶对齐
  45. {
  46. CDC *pDC = CDC::FromHandle(lpDrawItemStruct->hDC);
  47. CRect crect;
  48. GetWindowRect(&crect);
  49. CSize size = pDC->GetTextExtent(strCaption);
  50. if (X_RIGHT == m_xAlignment)    // 右对齐
  51. {
  52. x = crect.Width() - size.cx;
  53. }
  54. else if (X_CENTER == m_xAlignment)   // X居中对齐
  55. {
  56. x = (crect.Width()- size.cx) / 2;
  57. }
  58. if (Y_BOTTOM == m_yAlignment)   // 顶对齐
  59. {
  60. y = crect.Height() - size.cy;
  61. }
  62. else if (Y_CENTER == m_yAlignment)   // Y居中对齐
  63. {
  64. y = (crect.Height() - size.cy) / 2;
  65. }
  66. }
  67. // 设置dc字串颜色
  68. ::SetTextColor(lpDrawItemStruct->hDC, m_clrText);
  69. TextOut(lpDrawItemStruct->hDC, x, y, strCaption, strCaption.GetLength());
  70. }
  71. void CRichStatic::PreSubclassWindow()
  72. {
  73. CStatic::PreSubclassWindow();
  74. ModifyStyle(0, SS_OWNERDRAW);
  75. }
  76. void CRichStatic::SetFont(CString strFaceName, LONG nHeight)
  77. {
  78. if (m_selfCreated && m_hFont != NULL)
  79. {
  80. DeleteObject(m_hFont);    // 若字体对象为对象自己创建并且不为NULL,则销毁掉以释放内核对象
  81. }
  82. CFont cfont;
  83. LOGFONT lf;
  84. memset(&lf, 0, sizeof lf);    // 清空LOGFONT结构体,之后对其赋值
  85. lf.lfHeight = nHeight;
  86. _tcscpy_s(lf.lfFaceName, strFaceName.GetBuffer());    // 将字体名拷贝到LOGFONT结构体中
  87. VERIFY(cfont.CreateFontIndirect(&lf));    // 创建新的字体
  88. m_hFont = (HFONT)cfont.m_hObject;
  89. m_selfCreated = TRUE;    // 标记字体为自己创建的
  90. }
  91. void CRichStatic::SetFont(HFONT hFont)
  92. {
  93. if (m_selfCreated && m_hFont != NULL)
  94. {
  95. DeleteObject(m_hFont);    // 若字体对象为对象自己创建并且不为NULL,则销毁掉以释放内核对象
  96. }
  97. m_hFont = hFont;
  98. m_selfCreated = FALSE;   // 标记字体非自己创建
  99. }
  100. void CRichStatic::SetFont(const CFont *pFont)
  101. {
  102. if (m_selfCreated && m_hFont != NULL)
  103. {
  104. DeleteObject(m_hFont);    // 若字体对象为对象自己创建并且不为NULL,则销毁掉以释放内核对象
  105. }
  106. m_hFont = (HFONT)pFont->m_hObject;
  107. m_selfCreated = FALSE;   // 标记字体非自己创建
  108. }
  109. BOOL CRichStatic::OnEraseBkgnd(CDC* pDC)
  110. {
  111. // 当背景色为透明时,需要保存与拷贝显示主框的显示区域
  112. if (m_clrBackground == CLR_NONE)
  113. {
  114. if (m_Bmp.GetSafeHandle() == NULL)
  115. {
  116. CRect Rect;
  117. GetWindowRect(&Rect);
  118. CWnd *pParent = GetParent();
  119. ASSERT(pParent);
  120. pParent->ScreenToClient(&Rect);  // 将坐标转换为与主对话框相对应
  121. // 拷贝对应区域主框显示的内容
  122. CDC *pDC = pParent->GetDC();
  123. CDC MemDC;
  124. MemDC.CreateCompatibleDC(pDC);
  125. m_Bmp.CreateCompatibleBitmap(pDC,Rect.Width(),Rect.Height());
  126. CBitmap *pOldBmp = MemDC.SelectObject(&m_Bmp);
  127. MemDC.BitBlt(0,0,Rect.Width(),Rect.Height(),pDC,Rect.left,Rect.top,SRCCOPY);
  128. MemDC.SelectObject(pOldBmp);
  129. MemDC.DeleteDC();    // 删除内存DC,否则内存泄漏
  130. pParent->ReleaseDC(pDC);
  131. }
  132. else // 将主框显示的内容拷贝回去
  133. {
  134. CRect Rect;
  135. GetClientRect(Rect);
  136. CDC MemDC;
  137. MemDC.CreateCompatibleDC(pDC);
  138. CBitmap *pOldBmp = MemDC.SelectObject(&m_Bmp);
  139. pDC->BitBlt(0,0,Rect.Width(),Rect.Height(),&MemDC,0,0,SRCCOPY);
  140. MemDC.SelectObject(pOldBmp);
  141. MemDC.DeleteDC();    // 删除内存DC,否则内存泄漏
  142. }
  143. }
  144. return TRUE;
  145. }
  146. LRESULT CRichStatic::OnSetText(WPARAM wParam,LPARAM lParam)
  147. {
  148. LRESULT Result = Default();
  149. Invalidate();
  150. UpdateWindow();
  151. return Result;
  152. }
from:http://blog.csdn.net/cashey1991/article/details/7545614
04-15 16:59