PCWnd.cpp
上传用户:xajzjg
上传日期:2007-01-01
资源大小:277k
文件大小:8k
源码类别:

Windows编程

开发平台:

Visual C++

  1. // PCWnd.cpp : implementation file
  2. //
  3. #include "stdafx.h"
  4. #include "PCWnd.h"
  5. #ifdef _DEBUG
  6. #define new DEBUG_NEW
  7. #undef THIS_FILE
  8. static char THIS_FILE[] = __FILE__;
  9. #endif
  10. /////////////////////////////////////////////////////////////////////////////
  11. // CPCWnd
  12. CPCWnd::CPCWnd()
  13. {
  14. m_pOuter = NULL;
  15. }
  16. CPCWnd::~CPCWnd()
  17. {
  18. }
  19. BOOL CPCWnd::Create(CPopChildWnd* pOuter, DWORD dwStyle, const RECT& rect, HWND hParentWnd, UINT nID)
  20. {
  21. ASSERT(pOuter != NULL);
  22. BOOL bRet = CreateEx(NULL, AfxRegisterWndClass(0), _T(""), dwStyle, 
  23. rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, 
  24. hParentWnd, NULL, NULL );
  25. m_pOuter = pOuter;
  26. return bRet;
  27. }
  28. BEGIN_MESSAGE_MAP(CPCWnd, CWnd)
  29. //{{AFX_MSG_MAP(CPCWnd)
  30. ON_WM_MOUSEMOVE()
  31. ON_WM_LBUTTONDOWN()
  32. ON_WM_LBUTTONUP()
  33. ON_WM_PAINT()
  34. ON_WM_ERASEBKGND()
  35. ON_WM_SETCURSOR()
  36. //}}AFX_MSG_MAP
  37. END_MESSAGE_MAP()
  38. /////////////////////////////////////////////////////////////////////////////
  39. // CPCWnd message handlers
  40. void CPCWnd::OnMouseMove(UINT nFlags, CPoint point) 
  41. {
  42. if(m_pOuter->OnMouseMove(this))
  43. ;
  44. else
  45. {
  46. if(m_pOuter->Captured())
  47. {
  48. CPoint point;
  49. ::GetCursorPos(&point);
  50. if(GetStyle() & WS_CHILD)
  51. GetParent()->ScreenToClient(&point);
  52. SetWindowPos(NULL, point.x-m_point.x, point.y-m_point.y, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  53. }
  54. }
  55. CWnd::OnMouseMove(nFlags, point);
  56. }
  57. void CPCWnd::OnLButtonDown(UINT nFlags, CPoint point) 
  58. {
  59. if(m_pOuter->OnLButtonDown(this, point))
  60. ;
  61. else
  62. {
  63. SetCapture();
  64. m_point = point;
  65. }
  66. CWnd::OnLButtonDown(nFlags, point);
  67. }
  68. void CPCWnd::OnLButtonUp(UINT nFlags, CPoint point) 
  69. {
  70. if(GetCapture() == this)
  71. {
  72. ReleaseCapture();
  73. }
  74. m_pOuter->OnLButtonUp(this);
  75. CWnd::OnLButtonUp(nFlags, point);
  76. }
  77. BOOL CPCWnd::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)
  78. {
  79. if(m_pOuter->OnSetCursor(this, nHitTest, message))
  80. return TRUE;
  81. return CWnd::OnSetCursor(pWnd, nHitTest, message);
  82. }
  83. void CPCWnd::OnPaint() 
  84. {
  85. CPaintDC dc(this); // device context for painting
  86. Draw(&dc);
  87. }
  88. BOOL CPCWnd::OnEraseBkgnd(CDC* pDC) 
  89. {
  90. if(m_pOuter == NULL)
  91. return FALSE;
  92. CBrush backBrush(m_pOuter->OnGetBkColor());
  93. CBrush* pOldBrush = pDC->SelectObject(&backBrush);
  94. CRect rect;
  95. pDC->GetClipBox(&rect); //erase the area needed
  96. pDC->PatBlt(rect.left,rect.top,rect.Width(),rect.Height(),PATCOPY);
  97. pDC->SelectObject(pOldBrush);
  98. return TRUE;
  99. }
  100. void CPCWnd::Draw(CDC* pDC)
  101. {
  102. m_pOuter->OnDraw(pDC, this);
  103. }
  104. ///////////////////////////////////////////////////////////
  105. // CPopChildWnd
  106. CPopChildWnd::CPopChildWnd()
  107. {
  108. m_BkColor = RGB(129, 129, 129);
  109. }
  110. CPopChildWnd::~CPopChildWnd()
  111. {
  112. }
  113. void CPopChildWnd::Create(HWND hParentOwner, CRect& rect, BOOL bChildFirst)
  114. {
  115. DWORD styleChild, stylePop;
  116. if(bChildFirst)
  117. {
  118. styleChild = WS_CHILD | WS_VISIBLE;
  119. stylePop = WS_POPUP;
  120. }
  121. else
  122. {
  123. styleChild = WS_CHILD;
  124. stylePop = WS_POPUP | WS_VISIBLE;
  125. }
  126. #define CHILDID 200
  127. m_ChildWnd.Create(this, styleChild, rect, hParentOwner, CHILDID);
  128. m_PopupWnd.Create(this, stylePop, rect, hParentOwner, CHILDID+1);
  129. m_PopupWnd.Invalidate();
  130. m_hParentOwner = hParentOwner;
  131. }
  132. ////////////////////////////////
  133. // return FALSE to let the window to continue its handling
  134. //
  135. BOOL CPopChildWnd::OnMouseMove(CPCWnd* pWnd)
  136. {
  137. if(CWnd::GetCapture() != &m_ChildWnd && CWnd::GetCapture() != &m_PopupWnd)
  138. return FALSE;
  139. return OnChange(pWnd);
  140. }
  141. BOOL CPopChildWnd::Captured()
  142. {
  143. return (CWnd::GetCapture() == &m_ChildWnd || CWnd::GetCapture() == &m_PopupWnd);
  144. }
  145. BOOL CPopChildWnd::OnChange(CPCWnd* pWhichWnd)
  146. {
  147. CRect rcWnd, rcParent;
  148. CPCWnd* pWnd;
  149. if(m_ChildWnd.GetStyle() & WS_VISIBLE)
  150. pWnd = &m_ChildWnd;
  151. else
  152. pWnd = &m_PopupWnd;
  153. pWnd->GetWindowRect(&rcWnd);
  154. ::GetWindowRect(m_hParentOwner, &rcParent);
  155. rcWnd.NormalizeRect();
  156. rcParent.NormalizeRect();
  157. int iRet = RectRect(rcParent, rcWnd);
  158. if(pWnd->GetStyle() & WS_CHILD) // child window
  159. {
  160. if(iRet == TOUCH || iRet == OUTSIDE)  // switch to popup
  161. {
  162. m_ChildWnd.ShowWindow(SW_HIDE);
  163. CPoint point;
  164. ::GetCursorPos(&point);
  165. m_PopupWnd.SetWindowPos(NULL, point.x - m_offPt.x, point.y-m_offPt.y, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  166. m_PopupWnd.ShowWindow(SW_SHOW);
  167. return TRUE;
  168. }
  169. if(iRet == INSIDE) //
  170. {
  171. CPoint point;
  172. ::GetCursorPos(&point);
  173. ::ScreenToClient(m_hParentOwner, &point);
  174. m_ChildWnd.SetWindowPos(NULL, point.x-m_offPt.x, point.y-m_offPt.y, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  175. return TRUE;
  176. }
  177. }
  178. else // popup window
  179. {
  180. if(iRet == INSIDE)  // switch to child
  181. {
  182. m_PopupWnd.ShowWindow(SW_HIDE);
  183.   CPoint point;
  184. ::GetCursorPos(&point);
  185. ::ScreenToClient(m_hParentOwner, &point);
  186. m_ChildWnd.SetWindowPos(NULL, point.x-m_offPt.x, point.y-m_offPt.y, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  187. m_ChildWnd.ShowWindow(SW_SHOW);
  188. return TRUE;
  189. }
  190. if(iRet == TOUCH || iRet == OUTSIDE)  // just switched from child
  191. {
  192. CPoint point;
  193. ::GetCursorPos(&point);
  194. m_PopupWnd.SetWindowPos(NULL, point.x-m_offPt.x, point.y-m_offPt.y, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  195. return TRUE;
  196. }
  197. }
  198. return FALSE;
  199. }
  200. BOOL CPopChildWnd::OnLButtonDown(CPCWnd* pWnd, CPoint point)
  201. {
  202. m_offPt = point;
  203. return FALSE;
  204. }
  205. BOOL CPopChildWnd::OnLButtonUp(CPCWnd* pWnd)
  206. {
  207. return OnChange(NULL);
  208. }
  209. //////////////////////////////////////////////////////////////
  210. // Override this function to provide custom drawing.
  211. //
  212. void CPopChildWnd::OnDraw(CDC* pDC, CPCWnd* pWnd)
  213. {
  214. CString str = "Draw your stuff here";
  215. pDC->SetBkMode(TRANSPARENT);
  216. pDC->TextOut(0, 0, str);
  217. }
  218. ///////////////////////////////////////////
  219. // Override this function to set a cursor when mouse moves into this window.
  220. //
  221. BOOL CPopChildWnd::OnSetCursor(CPCWnd* pWnd, UINT nHitTest, UINT message)
  222. {
  223. return FALSE;
  224. }
  225. void CPopChildWnd::Destroy()
  226. {
  227. m_ChildWnd.DestroyWindow();
  228. m_PopupWnd.DestroyWindow();
  229. }
  230. /////////////////////
  231. // Force a switch
  232. //
  233. void CPopChildWnd::Switch()
  234. {
  235. CRect rcWnd;
  236. if(m_ChildWnd.GetStyle() & WS_VISIBLE)
  237. {
  238. m_ChildWnd.GetWindowRect(&rcWnd);
  239. m_ChildWnd.ShowWindow(SW_HIDE);
  240. m_PopupWnd.SetWindowPos(NULL, rcWnd.left, rcWnd.top, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  241. m_PopupWnd.ShowWindow(SW_SHOW);
  242. }
  243. else
  244. {
  245. m_PopupWnd.GetWindowRect(&rcWnd);
  246. CPoint point(rcWnd.left, rcWnd.top);
  247. ::ScreenToClient(m_hParentOwner, &point);
  248. m_PopupWnd.ShowWindow(SW_HIDE);
  249. m_ChildWnd.SetWindowPos(NULL, point.x, point.y, 0, 0, SWP_NOSIZE | SWP_NOACTIVATE);
  250. m_ChildWnd.ShowWindow(SW_SHOW);
  251. }
  252. }
  253. //////////////////////////////
  254. // When setting position, if current visible window is popup, the rect
  255. // is in screen coordinates. If it's the child window, the rect is in
  256. // the parent-owner's client coordinates.
  257. //
  258. void CPopChildWnd::SetPos(CRect& rect, BOOL bChildOnly)
  259. {
  260. if(!::IsWindow(m_ChildWnd.GetSafeHwnd()) || !::IsWindow(m_PopupWnd.GetSafeHwnd()))
  261. return;
  262. if(m_ChildWnd.GetStyle() & WS_VISIBLE)
  263. {
  264. m_ChildWnd.MoveWindow(rect);
  265. }
  266. else
  267. if(!bChildOnly)
  268. m_PopupWnd.MoveWindow(rect);
  269. }
  270. ///////////////////////////////////////////////////////////////
  271. // Tests the condition of two rects. Returns INSIDE if rc2 is complete inside
  272. // of rc1, OUTSIDE if rc2 is completely outside of rc1, and TOUCH otherwise.
  273. //
  274. int CPopChildWnd::RectRect(CRect rc1, CRect rc2)
  275. {
  276. if(rc2.left > rc1.left &&
  277. rc2.top > rc1.top &&
  278. rc2.right < rc1.right &&
  279. rc2.bottom < rc1.bottom)
  280. return INSIDE;
  281. if(rc2.bottom < rc1.top ||
  282. rc2.top > rc1.bottom ||
  283. rc2.right < rc1.left ||
  284. rc2.left > rc1.right)
  285. return OUTSIDE;
  286. return TOUCH;
  287. }