ROUNDBUTTON.CPP
上传用户:sjsmail
上传日期:2009-08-28
资源大小:260k
文件大小:10k
源码类别:

棋牌游戏

开发平台:

Visual C++

  1. // RoundButton.cpp : implementation file
  2. //
  3. // Round Buttons!
  4. //
  5. // Written by Chris Maunder (chrismaunder@codeguru.com)
  6. // Copyright (c) 1997,1998.
  7. // 
  8. // Modified: 2 Feb 1998 - Fix vis problem, CRgn resource leak,
  9. //                        button reposition code redone. CJM.
  10. //
  11. // This code may be used in compiled form in any way you desire. This
  12. // file may be redistributed unmodified by any means PROVIDING it is 
  13. // not sold for profit without the authors written consent, and 
  14. // providing that this notice and the authors name is included. If 
  15. // the source code in this file is used in any commercial application 
  16. // then a simple email would be nice.
  17. //
  18. // This file is provided "as is" with no expressed or implied warranty.
  19. // The author accepts no liability if it causes any damage to your
  20. // computer, causes your pet cat to fall ill, increases baldness or
  21. // makes you car start emitting strange noises when you start it up.
  22. //
  23. // Expect bugs.
  24. // 
  25. // Please use and enjoy. Please let me know of any bugs/mods/improvements 
  26. // that you have found/implemented and I will fix/incorporate them into this
  27. // file. 
  28. //
  29. /////////////////////////////////////////////////////////////////////////////
  30. #include "stdafx.h"
  31. #include "math.h"
  32. #include "RoundButton.h"
  33. #ifdef _DEBUG
  34. #define new DEBUG_NEW
  35. #undef THIS_FILE
  36. static char THIS_FILE[] = __FILE__;
  37. #endif
  38. // prototypes
  39. COLORREF GetColour(double dAngle, COLORREF crBright, COLORREF crDark);
  40. void DrawCircle(CDC* pDC, CPoint p, LONG lRadius, COLORREF crColour, BOOL bDashed = FALSE);
  41. void DrawCircle(CDC* pDC, CPoint p, LONG lRadius, COLORREF crBright, COLORREF crDark);
  42. // Calculate colour for a point at the given angle by performing a linear
  43. // interpolation between the colours crBright and crDark based on the cosine
  44. // of the angle between the light source and the point.
  45. //
  46. // Angles are measured from the +ve x-axis (i.e. (1,0) = 0 degrees, (0,1) = 90 degrees )
  47. // But remember: +y points down!
  48. COLORREF GetColour(double dAngle, COLORREF crBright, COLORREF crDark)
  49. {
  50. #define Rad2Deg 180.0/3.1415 
  51. #define LIGHT_SOURCE_ANGLE -2.356 // -2.356 radians = -135 degrees, i.e. From top left
  52. ASSERT(dAngle > -3.1416 && dAngle < 3.1416);
  53. double dAngleDifference = LIGHT_SOURCE_ANGLE - dAngle;
  54. if (dAngleDifference < -3.1415) dAngleDifference = 6.293 + dAngleDifference;
  55. else if (dAngleDifference > 3.1415) dAngleDifference = 6.293 - dAngleDifference;
  56. double Weight = 0.5*(cos(dAngleDifference)+1.0);
  57. BYTE Red   = (BYTE) (Weight*GetRValue(crBright) + (1.0-Weight)*GetRValue(crDark));
  58. BYTE Green = (BYTE) (Weight*GetGValue(crBright) + (1.0-Weight)*GetGValue(crDark));
  59. BYTE Blue  = (BYTE) (Weight*GetBValue(crBright) + (1.0-Weight)*GetBValue(crDark));
  60. //TRACE("LightAngle = %0.0f, Angle = %3.0f, Diff = %3.0f, Weight = %0.2f, RGB %3d,%3d,%3dn", 
  61. //   LIGHT_SOURCE_ANGLE*Rad2Deg, dAngle*Rad2Deg, dAngleDifference*Rad2Deg, Weight,Red,Green,Blue);
  62. return RGB(Red, Green, Blue);
  63. }
  64. void DrawCircle(CDC* pDC, CPoint p, LONG lRadius, COLORREF crColour, BOOL bDashed)
  65. {
  66. const int nDashLength = 1;
  67. LONG lError, lXoffset, lYoffset;
  68. int  nDash = 0;
  69. BOOL bDashOn = TRUE;
  70. //Check to see that the coordinates are valid
  71. ASSERT( (p.x + lRadius <= LONG_MAX) && (p.y + lRadius <= LONG_MAX) );
  72. ASSERT( (p.x - lRadius >= LONG_MIN) && (p.y - lRadius >= LONG_MIN) );
  73. //Set starting values
  74. lXoffset = lRadius;
  75. lYoffset = 0;
  76. lError   = -lRadius;
  77. do {
  78. if (bDashOn) {
  79. pDC->SetPixelV(p.x + lXoffset, p.y + lYoffset, crColour);
  80. pDC->SetPixelV(p.x + lXoffset, p.y - lYoffset, crColour);
  81. pDC->SetPixelV(p.x + lYoffset, p.y + lXoffset, crColour);
  82. pDC->SetPixelV(p.x + lYoffset, p.y - lXoffset, crColour);
  83. pDC->SetPixelV(p.x - lYoffset, p.y + lXoffset, crColour);
  84. pDC->SetPixelV(p.x - lYoffset, p.y - lXoffset, crColour);
  85. pDC->SetPixelV(p.x - lXoffset, p.y + lYoffset, crColour);
  86. pDC->SetPixelV(p.x - lXoffset, p.y - lYoffset, crColour);
  87. }
  88. //Advance the error term and the constant X axis step
  89. lError += lYoffset++;
  90. //Check to see if error term has overflowed
  91. if ((lError += lYoffset) >= 0)
  92. lError -= --lXoffset * 2;
  93. if (bDashed && (++nDash == nDashLength)) {
  94. nDash = 0;
  95. bDashOn = !bDashOn;
  96. }
  97. } while (lYoffset <= lXoffset); //Continue until halfway point
  98. void DrawCircle(CDC* pDC, CPoint p, LONG lRadius, COLORREF crBright, COLORREF crDark)
  99. {
  100. LONG lError, lXoffset, lYoffset;
  101. //Check to see that the coordinates are valid
  102. ASSERT( (p.x + lRadius <= LONG_MAX) && (p.y + lRadius <= LONG_MAX) );
  103. ASSERT( (p.x - lRadius >= LONG_MIN) && (p.y - lRadius >= LONG_MIN) );
  104. //Set starting values
  105. lXoffset = lRadius;
  106. lYoffset = 0;
  107. lError   = -lRadius;
  108. do {
  109. const double Pi = 3.141592654, 
  110.  Pi_on_2 = Pi * 0.5,
  111.  Three_Pi_on_2 = Pi * 1.5;
  112. COLORREF crColour;
  113. double   dAngle = atan2(lYoffset, lXoffset);
  114. //Draw the current pixel, reflected across all eight arcs
  115. crColour = GetColour(dAngle, crBright, crDark);
  116. pDC->SetPixelV(p.x + lXoffset, p.y + lYoffset, crColour);
  117. crColour = GetColour(Pi_on_2 - dAngle, crBright, crDark);
  118. pDC->SetPixelV(p.x + lYoffset, p.y + lXoffset, crColour);
  119. crColour = GetColour(Pi_on_2 + dAngle, crBright, crDark);
  120. pDC->SetPixelV(p.x - lYoffset, p.y + lXoffset, crColour);
  121. crColour = GetColour(Pi - dAngle, crBright, crDark);
  122. pDC->SetPixelV(p.x - lXoffset, p.y + lYoffset, crColour);
  123. crColour = GetColour(-Pi + dAngle, crBright, crDark);
  124. pDC->SetPixelV(p.x - lXoffset, p.y - lYoffset, crColour);
  125. crColour = GetColour(-Pi_on_2 - dAngle, crBright, crDark);
  126. pDC->SetPixelV(p.x - lYoffset, p.y - lXoffset, crColour);
  127. crColour = GetColour(-Pi_on_2 + dAngle, crBright, crDark);
  128. pDC->SetPixelV(p.x + lYoffset, p.y - lXoffset, crColour);
  129. crColour = GetColour(-dAngle, crBright, crDark);
  130. pDC->SetPixelV(p.x + lXoffset, p.y - lYoffset, crColour);
  131. //Advance the error term and the constant X axis step
  132. lError += lYoffset++;
  133. //Check to see if error term has overflowed
  134. if ((lError += lYoffset) >= 0)
  135. lError -= --lXoffset * 2;
  136. } while (lYoffset <= lXoffset); //Continue until halfway point
  137. /////////////////////////////////////////////////////////////////////////////
  138. // CRoundButton
  139. CRoundButton::CRoundButton()
  140. {
  141. m_bDrawDashedFocusCircle = TRUE;
  142. }
  143. CRoundButton::~CRoundButton()
  144. {
  145. m_rgn.DeleteObject();
  146. }
  147. BEGIN_MESSAGE_MAP(CRoundButton, CButton)
  148. //{{AFX_MSG_MAP(CRoundButton)
  149. //}}AFX_MSG_MAP
  150. END_MESSAGE_MAP()
  151. /////////////////////////////////////////////////////////////////////////////
  152. // CRoundButton message handlers
  153. void CRoundButton::PreSubclassWindow() 
  154. {
  155. CButton::PreSubclassWindow();
  156. ModifyStyle(0, BS_OWNERDRAW);
  157. CRect rect;
  158. GetClientRect(rect);
  159. // Resize the window to make it square
  160. rect.bottom = rect.right = min(rect.bottom,rect.right);
  161. // Get the vital statistics of the window
  162. m_ptCentre = rect.CenterPoint();
  163. m_nRadius  = rect.bottom/2-1;
  164. // Set the window region so mouse clicks only activate the round section 
  165. // of the button
  166. m_rgn.DeleteObject(); 
  167. SetWindowRgn(NULL, FALSE);
  168. m_rgn.CreateEllipticRgnIndirect(rect);
  169. SetWindowRgn(m_rgn, TRUE);
  170. // Convert client coords to the parents client coords
  171. ClientToScreen(rect);
  172. CWnd* pParent = GetParent();
  173. if (pParent) pParent->ScreenToClient(rect);
  174. // Resize the window
  175. MoveWindow(rect.left, rect.top, rect.Width(), rect.Height(), TRUE);
  176. }
  177. void CRoundButton::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct) 
  178. {
  179. ASSERT(lpDrawItemStruct != NULL);
  180. CDC* pDC   = CDC::FromHandle(lpDrawItemStruct->hDC);
  181. CRect rect = lpDrawItemStruct->rcItem;
  182. UINT state = lpDrawItemStruct->itemState;
  183. UINT nStyle = GetStyle();
  184. int nRadius = m_nRadius;
  185. int nSavedDC = pDC->SaveDC();
  186. pDC->SelectStockObject(NULL_BRUSH);
  187. pDC->FillSolidRect(rect, ::GetSysColor(COLOR_BTNFACE));
  188. // Draw the focus circle around the button
  189. if ((state & ODS_FOCUS) && m_bDrawDashedFocusCircle)
  190. DrawCircle(pDC, m_ptCentre, nRadius--, RGB(0,0,0));
  191. // Draw the raised/sunken edges of the button (unless flat)
  192. if (nStyle & BS_FLAT) {
  193. DrawCircle(pDC, m_ptCentre, nRadius--, RGB(0,0,0));
  194. DrawCircle(pDC, m_ptCentre, nRadius--, ::GetSysColor(COLOR_3DHIGHLIGHT));
  195. } else {
  196. if ((state & ODS_SELECTED)) {
  197. DrawCircle(pDC, m_ptCentre, nRadius--, 
  198.    ::GetSysColor(COLOR_3DDKSHADOW), ::GetSysColor(COLOR_3DHIGHLIGHT));
  199. DrawCircle(pDC, m_ptCentre, nRadius--, 
  200.    ::GetSysColor(COLOR_3DSHADOW), ::GetSysColor(COLOR_3DLIGHT));
  201. } else {
  202. DrawCircle(pDC, m_ptCentre, nRadius--, 
  203.    ::GetSysColor(COLOR_3DHIGHLIGHT), ::GetSysColor(COLOR_3DDKSHADOW));
  204. DrawCircle(pDC, m_ptCentre, nRadius--, 
  205.    ::GetSysColor(COLOR_3DLIGHT), ::GetSysColor(COLOR_3DSHADOW));
  206. }
  207. }
  208. // draw the text if there is any
  209. CString strText;
  210. GetWindowText(strText);
  211. if (!strText.IsEmpty())
  212. {
  213. CRgn rgn;
  214. rgn.CreateEllipticRgn(m_ptCentre.x-nRadius, m_ptCentre.y-nRadius, 
  215.   m_ptCentre.x+nRadius, m_ptCentre.y+nRadius);
  216. pDC->SelectClipRgn(&rgn);
  217. CSize Extent = pDC->GetTextExtent(strText);
  218. CPoint pt = CPoint( m_ptCentre.x - Extent.cx/2, m_ptCentre.x - Extent.cy/2 );
  219. if (state & ODS_SELECTED) pt.Offset(1,1);
  220. pDC->SetBkMode(TRANSPARENT);
  221. if (state & ODS_DISABLED)
  222. pDC->DrawState(pt, Extent, strText, DSS_DISABLED, TRUE, 0, (HBRUSH)NULL);
  223. else
  224. pDC->TextOut(pt.x, pt.y, strText);
  225. pDC->SelectClipRgn(NULL);
  226. rgn.DeleteObject();
  227. }
  228. // Draw the focus circle on the inside of the button
  229. if ((state & ODS_FOCUS) && m_bDrawDashedFocusCircle)
  230. DrawCircle(pDC, m_ptCentre, nRadius-2, RGB(0,0,0), TRUE);
  231. pDC->RestoreDC(nSavedDC);
  232. }