ButtonExt.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. // ButtonExt.cpp: implementation of the CButtonExt class.
  2. //
  3. //////////////////////////////////////////////////////////////////////
  4. #include "stdafx.h"
  5. #include "ButtonExt.h"
  6. #ifdef _DEBUG
  7. #undef THIS_FILE
  8. static char THIS_FILE[]=__FILE__;
  9. #define new DEBUG_NEW
  10. #endif
  11. UINT CButtonExt::m_nStatus = STATUS_BUTTON_NORMAL;
  12. WNDPROC CButtonExt::m_cWndProc = NULL;
  13. CArray<int,int>g_btnarray;
  14. //按钮的状态
  15. BOOL m_bOver=0; //鼠标位于按钮之上时该值为true,反之为flase
  16. BOOL m_bTracking=0; //在鼠标按下没有释放时该值为true
  17. BOOL m_bSelected=0; //按钮被按下是该值为true
  18. BOOL m_bFocus=0; //按钮为当前焦点所在时该值为true
  19. //按钮的外边框
  20. extern CPen m_BoundryPen;
  21. //鼠标指针置于按钮之上时按钮的内边框
  22. extern CPen m_InsideBoundryPenLeft;
  23. extern CPen m_InsideBoundryPenRight;
  24. extern CPen m_InsideBoundryPenTop;
  25. extern CPen m_InsideBoundryPenBottom;
  26. //按钮获得焦点时按钮的内边框
  27. extern CPen m_InsideBoundryPenLeftSel;
  28. extern CPen m_InsideBoundryPenRightSel;
  29. extern CPen m_InsideBoundryPenTopSel;
  30. extern CPen m_InsideBoundryPenBottomSel;
  31. //按钮的底色,包括有效和无效两种状态
  32. extern CBrush m_FillActive;
  33. extern CBrush m_FillInactive;
  34. //绘制按钮的底色
  35. void CButtonExt::DoGradientFill(CDC *pDC, CRect* rect)
  36. {return;
  37. CBrush brBk[64];
  38. int nWidth = rect->Width();
  39. int nHeight = rect->Height();
  40. CRect rct;
  41. for (int i = 0; i < 64; i ++)
  42. {
  43. if (m_bOver)
  44. {
  45. if (m_bFocus)
  46. brBk[i].CreateSolidBrush(RGB(255 - (i / 4), 255 - (i / 4), 255 - (i / 3)));
  47. else
  48. brBk[i].CreateSolidBrush(RGB(255 - (i / 4), 255 - (i / 4), 255 - (i / 5)));
  49. }
  50. else
  51. {
  52. if (m_bFocus)
  53. brBk[i].CreateSolidBrush(RGB(255 - (i / 3), 255 - (i / 3), 255 - (i / 4)));
  54. else
  55. brBk[i].CreateSolidBrush(RGB(255 - (i / 3), 255 - (i / 3), 255 - (i / 5)));
  56. }
  57. }
  58. for (i = rect->top; i <= nHeight + 2; i ++)
  59. {
  60. rct.SetRect(rect->left, i, nWidth + 2, i + 1);
  61. pDC->FillRect(&rct, &brBk[((i * 63) / nHeight)]);
  62. }
  63. for (i = 0; i < 64; i ++)
  64. brBk[i].DeleteObject();
  65. }
  66. //绘制按钮的内边框
  67. void CButtonExt::DrawInsideBorder(CDC *pDC, CRect* rect)
  68. {
  69. CPen *pLeft, *pRight, *pTop, *pBottom;
  70. if (m_bSelected && !m_bOver)
  71. {
  72. pLeft = & m_InsideBoundryPenLeftSel;
  73. pRight = &m_InsideBoundryPenRightSel;
  74. pTop = &m_InsideBoundryPenTopSel;
  75. pBottom = &m_InsideBoundryPenBottomSel;
  76. }
  77. else
  78. {
  79. pLeft = &m_InsideBoundryPenLeft;
  80. pRight = &m_InsideBoundryPenRight;
  81. pTop = &m_InsideBoundryPenTop;
  82. pBottom = &m_InsideBoundryPenBottom;
  83. }
  84. CPoint oldPoint = pDC->MoveTo(rect->left, rect->bottom - 1);
  85. CPen* pOldPen = pDC->SelectObject(pLeft);
  86. pDC->LineTo(rect->left, rect->top + 1);
  87. pDC->SelectObject(pRight);
  88. pDC->MoveTo(rect->right - 1, rect->bottom - 1);
  89. pDC->LineTo(rect->right - 1, rect->top);
  90. pDC->SelectObject(pTop);
  91. pDC->MoveTo(rect->left - 1, rect->top);
  92. pDC->LineTo(rect->right - 1, rect->top);
  93. pDC->SelectObject(pBottom);
  94. pDC->MoveTo(rect->left, rect->bottom);
  95. pDC->LineTo(rect->right - 1, rect->bottom);
  96. pDC->SelectObject(pOldPen);
  97. pDC->MoveTo(oldPoint);
  98. if (m_bSelected && !m_bOver)
  99. DrawFocusRect(pDC->m_hDC,rect);
  100. }
  101. //////////////////////////////////////////////////////////////////////
  102. // Construction/Destruction
  103. //////////////////////////////////////////////////////////////////////
  104. int FindBtnPos(int hwnd)
  105. {
  106. for(int i=0; i<g_btnarray.GetSize (); i++)
  107. {
  108. if(hwnd==g_btnarray.ElementAt (i))return i;
  109. }
  110. return -1;
  111. }
  112. LRESULT CButtonExt::DefWindowProc( HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam )
  113. {
  114. CWnd *pWnd = NULL;
  115. CPoint point;
  116. pWnd = CWnd::FromHandle( hWnd );
  117. switch( message )
  118. {
  119. case WM_CREATE:
  120. g_btnarray.Add ((int)hWnd);
  121. g_btnarray.Add (0);
  122. g_btnarray.Add (0);
  123. g_btnarray.Add (0);
  124. g_btnarray.Add (0);
  125. break;
  126. case WM_DESTROY:
  127. {
  128. int pos=FindBtnPos((int)hWnd);
  129. g_btnarray.RemoveAt(pos);
  130. g_btnarray.RemoveAt(pos);
  131. g_btnarray.RemoveAt(pos);
  132. g_btnarray.RemoveAt(pos);
  133. g_btnarray.RemoveAt(pos);
  134. }
  135. break;
  136. case WM_MOUSEMOVE:
  137. {
  138. int pos=FindBtnPos((int)hWnd);
  139. m_bTracking=g_btnarray.ElementAt (pos+2);
  140. if (!m_bTracking)
  141. {
  142. TRACKMOUSEEVENT tme;
  143. tme.cbSize = sizeof(tme);
  144. tme.hwndTrack = hWnd;
  145. tme.dwFlags = TME_LEAVE | TME_HOVER;
  146. tme.dwHoverTime = 1;
  147. m_bTracking = _TrackMouseEvent(&tme);
  148. g_btnarray.SetAt (pos+2, m_bTracking);
  149. }
  150. }
  151. break;
  152. case WM_MOUSELEAVE:
  153. {
  154. int pos=FindBtnPos((int)hWnd);
  155. m_bOver = FALSE;
  156. m_bTracking = FALSE;
  157. g_btnarray.SetAt (pos+1, m_bOver);
  158. g_btnarray.SetAt (pos+2, m_bTracking);
  159. pWnd->Invalidate();
  160. }
  161. break;
  162. case WM_MOUSEHOVER:
  163. {
  164. int pos=FindBtnPos((int)hWnd);
  165. m_bOver = 1;
  166. g_btnarray.SetAt (pos+1, m_bOver);
  167. pWnd->Invalidate();
  168. }
  169. break;
  170. case WM_PAINT:
  171. // case WM_LBUTTONDBLCLK:
  172. return OnPaint( pWnd );
  173. break;
  174. /* case WM_LBUTTONDOWN:
  175. point.x = LOWORD(lParam);
  176. point.y = HIWORD(lParam);
  177. return OnLButtonDown( pWnd, 0, point );
  178. break;
  179. case WM_LBUTTONUP:
  180. point.x = LOWORD(lParam);
  181. point.y = HIWORD(lParam);
  182. return OnLButtonUp( pWnd, 0, point );
  183. break;
  184. case WM_MOUSEMOVE:
  185. point.x = LOWORD(lParam);
  186. point.y = HIWORD(lParam);
  187. return OnMouseMove( pWnd, 0, point );
  188. break;
  189. case WM_ERASEBKGND:
  190. return TRUE;
  191. break;
  192. case WM_ENABLE:
  193. {
  194. pWnd->Invalidate();
  195. return FALSE;
  196. }
  197. break;*/
  198. default:
  199. break;
  200. }
  201. return CallWindowProc( m_cWndProc, hWnd, message, wParam, lParam );
  202. }
  203. LRESULT CButtonExt::OnLButtonDown( CWnd *pWnd, UINT nFlags, CPoint point )
  204. {
  205. m_nStatus = STATUS_BUTTON_DOWN;
  206. // m_bOver=1; //鼠标位于按钮之上时该值为true,反之为flase
  207. // m_bTracking=1; //在鼠标按下没有释放时该值为true
  208. // m_bSelected=1; //按钮被按下是该值为true
  209. // m_bFocus=1; //按钮为当前焦点所在时该值为true
  210. pWnd->Invalidate();
  211. pWnd->UpdateWindow();
  212. return TRUE;
  213. }
  214. LRESULT CButtonExt::OnLButtonUp( CWnd *pWnd, UINT nFlags, CPoint point )
  215. {
  216. ReleaseCapture();
  217. m_nStatus = STATUS_BUTTON_NORMAL;
  218. pWnd->Invalidate();
  219. pWnd->UpdateWindow();
  220. SendMessage( pWnd->GetParent()->m_hWnd, WM_COMMAND, pWnd->GetDlgCtrlID(), (LPARAM) (pWnd->m_hWnd) );
  221. return TRUE;
  222. }
  223. LRESULT CButtonExt::OnMouseMove( CWnd *pWnd, UINT nFlags, CPoint point )
  224. {
  225. HRGN hRgn = CreateRectRgn( 0, 0, 0, 0 );
  226. pWnd->GetWindowRgn( hRgn );
  227. BOOL bIn = PtInRegion( hRgn, point.x, point.y );
  228. if( bIn )
  229. {
  230. if(CWnd::GetCapture()==pWnd)return 1;
  231. if( m_nStatus != STATUS_BUTTON_HOVER )
  232. {
  233. m_nStatus = STATUS_BUTTON_HOVER;
  234. pWnd->Invalidate();
  235. pWnd->UpdateWindow();
  236. pWnd->SetCapture();
  237. }
  238. }
  239. else
  240. {
  241. //if ( m_nStatus == STATUS_BUTTON_HOVER )
  242. //{
  243. m_nStatus = STATUS_BUTTON_NORMAL;
  244. pWnd->Invalidate();
  245. pWnd->UpdateWindow();
  246. ReleaseCapture();
  247. //}
  248. }
  249. DeleteObject( hRgn );
  250. return TRUE;
  251. }
  252. LRESULT CButtonExt::OnPaint( CWnd *pWnd )
  253. {
  254. CRect rect ;//= lpDrawItemStruct->rcItem;
  255. pWnd->GetClientRect( &rect );
  256. CPaintDC dc(pWnd);
  257. CDC *pDC=&dc;
  258. int nSaveDC=pDC->SaveDC();
  259. int pos=FindBtnPos((int)pWnd->GetSafeHwnd ());
  260. m_bOver=g_btnarray.ElementAt (pos+1);
  261. m_bTracking=g_btnarray.ElementAt (pos+2);
  262. m_bSelected=g_btnarray.ElementAt (pos+3);
  263. m_bFocus=g_btnarray.ElementAt (pos+4);
  264. POINT pt ;
  265. TCHAR strText[MAX_PATH + 1];
  266. ::GetWindowText(pWnd->GetSafeHwnd (), strText, MAX_PATH);
  267. //画按钮的外边框,它是一个半径为5的圆角矩形
  268. pt.x = 5;
  269. pt.y = 5;
  270. CPen* hOldPen = pDC->SelectObject(&m_BoundryPen);
  271. pDC->RoundRect(&rect, pt);
  272. pDC->SelectObject(hOldPen);
  273. rect.DeflateRect(CSize(GetSystemMetrics(SM_CXEDGE), GetSystemMetrics(SM_CYEDGE)));
  274. //根据按钮的状态填充按钮的底色
  275. CBrush* pOldBrush;
  276. if (m_bOver)
  277. {
  278. pOldBrush = pDC->SelectObject(&m_FillActive);
  279. DoGradientFill(pDC, &rect);
  280. }
  281. else
  282. {
  283. pOldBrush = pDC->SelectObject(&m_FillInactive);
  284. DoGradientFill(pDC, &rect);
  285. }
  286. //根据按钮的状态绘制内边框
  287. if (m_bOver || m_bSelected)
  288. DrawInsideBorder(pDC, &rect);
  289. pDC->SelectObject(pOldBrush);
  290. //显示按钮的文本
  291. if (strText!=NULL)
  292. {
  293. CFont* hFont = pWnd->GetFont();
  294. CFont* hOldFont = pDC->SelectObject(hFont);
  295. CSize szExtent = pDC->GetTextExtent(strText, lstrlen(strText));
  296. CPoint pt( rect.CenterPoint().x - szExtent.cx / 2, rect.CenterPoint().y - szExtent.cy / 2);
  297. short state=GetKeyState(VK_LBUTTON);
  298. BYTE sta=(BYTE)(state>>8);
  299. if (m_bTracking && sta)//state & ODS_SELECTED)
  300. pt.Offset(1, 1);
  301. int nMode = pDC->SetBkMode(TRANSPARENT);
  302. if (pWnd->IsWindowEnabled ()==0)
  303. pDC->DrawState(pt, szExtent, strText, DSS_DISABLED, TRUE, 0, (HBRUSH)NULL);
  304. else
  305. pDC->DrawState(pt, szExtent, strText, DSS_NORMAL, TRUE, 0, (HBRUSH)NULL);
  306. pDC->SelectObject(hOldFont);
  307. pDC->SetBkMode(nMode);
  308. }
  309. pDC->RestoreDC(nSaveDC);
  310. /* CPaintDC dc(pWnd);
  311. CString cs;
  312. RECT rc;
  313. CFont Font;
  314. CFont *pOldFont;
  315. CBrush Brush;
  316. CBrush *pOldBrush;
  317. CPen Pen;
  318. CPen *pOldPen;
  319. POINT pt;
  320. pt.x = 2;
  321. pt.y = 2;
  322. if(!pWnd->IsWindowEnabled ())m_nStatus = STATUS_BUTTON_DOWN;
  323. dc.SetBkMode( TRANSPARENT );
  324. Font.CreateFont( 14, 0, 0, 0, FW_HEAVY, 0, 0, 0, ANSI_CHARSET, \
  325. OUT_TT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, \
  326. VARIABLE_PITCH | FF_SWISS, "黑体" );
  327. pOldFont = dc.SelectObject( &Font );
  328. if( m_nStatus == STATUS_BUTTON_DOWN ) {
  329. Brush.CreateSolidBrush( RGB( 160, 160, 160 ) );
  330. Pen.CreatePen( PS_SOLID, 1, RGB( 100, 100, 100 ) );
  331. dc.SetTextColor( RGB( 50, 50, 250 ) );
  332. } else if( m_nStatus == STATUS_BUTTON_HOVER ) {
  333. Brush.CreateSolidBrush( RGB( 60, 60, 180 ) );
  334. Pen.CreatePen( PS_SOLID, 1, RGB( 0, 0, 0 ) );
  335. dc.SetTextColor( RGB( 250, 250, 50 ) );
  336. } else if( m_nStatus == STATUS_BUTTON_NORMAL )
  337. {
  338. Brush.CreateSolidBrush( RGB( 240, 240, 240 ) );
  339. Pen.CreatePen( PS_SOLID, 1, RGB( 120, 120, 120 ) );
  340. dc.SetTextColor( RGB( 50, 50, 50 ) );
  341. }
  342. pOldBrush = dc.SelectObject( &Brush );
  343. pOldPen = dc.SelectObject( &Pen );
  344. pWnd->GetClientRect( &rc );
  345. dc.RoundRect( &rc, pt );
  346. HRGN hRgn = CreateRectRgn( rc.left, rc.top, rc.right, rc.bottom );
  347. pWnd->SetWindowRgn( hRgn, TRUE );
  348. DeleteObject( hRgn );
  349. pWnd->GetWindowText( cs );
  350. dc.DrawText( cs, &rc, DT_CENTER | DT_VCENTER | DT_SINGLELINE );
  351. dc.SelectObject( pOldFont );
  352. dc.SelectObject( pOldBrush );
  353. dc.SelectObject( pOldPen );
  354. m_nStatus = STATUS_BUTTON_NORMAL;*/
  355. return TRUE;
  356. }