ButtonExt.cpp 10 KB

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