GFXPOPUPMENU.CPP
上传用户:maryhy001
上传日期:2007-05-02
资源大小:2317k
文件大小:15k
源码类别:

网格计算

开发平台:

Visual C++

  1. // GfxPopupMenu.cpp: implementation of the CGfxPopupMenu class.
  2. //
  3. //////////////////////////////////////////////////////////////////////
  4. #include "stdafx.h"
  5. #include "trfAgent.h"
  6. #include "GfxPopupMenu.h"
  7. //////////////////////////////////////////////////////////////////////
  8. // Construction/Destruction
  9. //////////////////////////////////////////////////////////////////////
  10. CGfxPopupMenu::CGfxPopupMenu()
  11. {
  12. crMenuText = GetSysColor(COLOR_MENUTEXT);
  13. crMenuTextSel = GetSysColor(COLOR_HIGHLIGHTTEXT);
  14. cr3dFace = GetSysColor(COLOR_3DFACE);
  15. crMenu = GetSysColor(COLOR_MENU);
  16. crHighlight = GetSysColor(COLOR_HIGHLIGHT);
  17. cr3dHilight = GetSysColor(COLOR_3DHILIGHT);
  18. cr3dShadow = GetSysColor(COLOR_3DSHADOW);
  19. crGrayText = GetSysColor(COLOR_GRAYTEXT);
  20. m_clrBtnFace = GetSysColor(COLOR_BTNFACE);
  21. m_clrBtnHilight = GetSysColor(COLOR_BTNHILIGHT);
  22. m_clrBtnShadow = GetSysColor(COLOR_BTNSHADOW);
  23. iSpawnItem = 0;
  24. pSpawnItem = NULL;
  25. iImageItem = 0;
  26. pImageItem = NULL;
  27. szImage = CSize(20,20);
  28. hMenuFont = NULL;
  29. /* COLORMAP cMap[3] = { 
  30. { RGB(128,128,128), cr3dShadow }, 
  31. { RGB(192,192,192), cr3dFace }, 
  32. { RGB(255,255,255), cr3dHilight }
  33. };
  34. CBitmap bmp;
  35. bmp.LoadMappedBitmap(IDB_MENUCHK, 0, cMap, 3);
  36. ilOther.Create(19, 19, ILC_COLOR4|ILC_MASK, 1, 0);
  37. ilOther.Add(&bmp, cr3dFace);
  38. bmp.DeleteObject();
  39. */
  40. NONCLIENTMETRICS ncm;
  41. memset(&ncm, 0, sizeof(ncm));
  42. ncm.cbSize = sizeof(ncm);
  43. ::SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, (PVOID) &ncm, 0);
  44. hGuiFont = ::CreateFontIndirect(&ncm.lfMenuFont);
  45. // David 08/04/98 - start - bold font handling
  46. hMenuBoldFont = NULL;
  47. CreateBoldFont();
  48. // David 08/04/98 - end - bold font handling
  49. }
  50. CGfxPopupMenu::~CGfxPopupMenu()
  51. {
  52. if (iSpawnItem > 0)
  53. {
  54. for (int t = 0; t < iSpawnItem; t++)
  55. if (pSpawnItem[t]) delete pSpawnItem[t];
  56. GlobalFree((HGLOBAL) pSpawnItem);
  57. }
  58. if (iImageItem > 0)
  59. {
  60. GlobalFree((HGLOBAL) pImageItem);
  61. }
  62. if (hGuiFont) ::DeleteObject((HGDIOBJ)hGuiFont);
  63. if (hMenuFont) ::DeleteObject((HGDIOBJ)hMenuFont);
  64. if (hMenuBoldFont) ::DeleteObject((HGDIOBJ)hMenuBoldFont);
  65. }
  66. void CGfxPopupMenu::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
  67. {
  68. // CDC* pDC = CDC::FromHandle(lpDrawItemStruct->hDC);
  69. // CRect rcItem(lpDrawItemStruct->rcItem);
  70. // pDC->FillSolidRect(rcItem, RGB(255,0,0));
  71. if (lpDrawItemStruct->CtlType == ODT_MENU)
  72. {
  73. UINT id = lpDrawItemStruct->itemID;
  74. UINT state = lpDrawItemStruct->itemState;
  75. bool bEnab = !(state & ODS_DISABLED);
  76. bool bSelect = (state & ODS_SELECTED) ? true : false;
  77. bool bChecked = (state & ODS_CHECKED) ? true : false;
  78. // David 08/04/98 - start - bold font handling
  79. bool bBold = (state & ODS_DEFAULT) ? true : false;
  80. // David 08/04/98 - end - bold font handling
  81. SpawnItem * pItem = (SpawnItem *) lpDrawItemStruct->itemData;
  82. if (pItem)
  83. {
  84. CDC * pDC = CDC::FromHandle(lpDrawItemStruct->hDC);
  85. CFont * pft;
  86. // David 08/04/98 - start - bold font handling
  87. if (!bBold) pft = CFont::FromHandle((HFONT) hMenuFont ? hMenuFont : hGuiFont);
  88. else pft = CFont::FromHandle((HFONT) hMenuBoldFont ? hMenuBoldFont : hGuiFont);
  89. // David 08/04/98 - end - bold font handling
  90. CFont * of = pDC->SelectObject(pft);
  91. CRect rc(lpDrawItemStruct->rcItem);
  92. CRect rcImage(rc), rcText(rc);
  93. rcImage.right = rcImage.left + rc.Height();
  94. rcImage.bottom = rc.bottom;
  95. if (pItem->iCmd == -3) // is a separator
  96. {
  97. CPen pnDk(PS_SOLID,1,cr3dShadow);
  98. CPen pnLt(PS_SOLID,1,cr3dHilight);
  99. CPen * opn = pDC->SelectObject(&pnDk);
  100. pDC->MoveTo(rc.left + 2, rc.top + 2);
  101. pDC->LineTo(rc.right - 2, rc.top + 2);
  102. pDC->SelectObject(&pnLt);
  103. pDC->MoveTo(rc.left + 2, rc.top + 3);
  104. pDC->LineTo(rc.right - 2, rc.top + 3);
  105. pDC->SelectObject(opn);
  106. }
  107. else if (pItem->iCmd == -4) // is a title item
  108. {
  109. CString cs(pItem->cText), cs1;
  110. CRect rcBdr(rcText);
  111. if (bSelect && bEnab)
  112. {
  113. rcText.top ++;
  114. rcText.left += 2;
  115. }
  116. pDC->FillSolidRect(rcText, crMenu);
  117. pDC->DrawText(cs, rcText, DT_VCENTER|DT_CENTER|DT_SINGLELINE);
  118. if (bSelect && bEnab) pDC->Draw3dRect(rcBdr,cr3dShadow,cr3dHilight);
  119. }
  120. else
  121. {
  122. rcText.left += rcImage.right + 1;
  123. int obk = pDC->SetBkMode(TRANSPARENT);
  124. COLORREF ocr;
  125. if (bSelect)
  126. {
  127. if (pItem->iImageIdx >= 0 || (state & ODS_CHECKED))
  128. pDC->FillSolidRect(rcText, crHighlight);
  129. else
  130. pDC->FillSolidRect(rc, crHighlight);
  131. ocr = pDC->SetTextColor(crMenuTextSel);
  132. }
  133. else
  134. {
  135. if (pItem->iImageIdx >= 0 || (state & ODS_CHECKED))
  136. pDC->FillSolidRect(rcText, crMenu);
  137. else
  138. pDC->FillSolidRect(rc/*rcText*/, crMenu);
  139. ocr = pDC->SetTextColor(crMenuText);
  140. }
  141. if (pItem->iImageIdx >= 0)
  142. {
  143. int ay = (rcImage.Height() - szImage.cy) / 2;
  144. int ax = (rcImage.Width()  - szImage.cx) / 2;
  145. if (bSelect && bEnab)
  146. pDC->Draw3dRect(rcImage,cr3dHilight,cr3dShadow);
  147. else
  148. {
  149. pDC->Draw3dRect(rcImage,crMenu,crMenu);
  150. }
  151. if (bEnab)
  152. {
  153. ilList.Draw(pDC, pItem->iImageIdx, CPoint(rcImage.left + ax, rcImage.top +ay), ILD_NORMAL);
  154. }
  155. else
  156. {
  157. HICON hIcon = ilList.ExtractIcon( pItem->iImageIdx );
  158. pDC->DrawState( CPoint(rcImage.left + ax, rcImage.top + ay ), szImage, (HICON)hIcon, DST_ICON | DSS_DISABLED, (CBrush *)NULL );
  159. }
  160. }
  161. else
  162. {
  163. if (bChecked)
  164. {
  165. int ay = (rcImage.Height() - szImage.cy) / 2;
  166. int ax = (rcImage.Width()  - szImage.cx) / 2;
  167. ilOther.Draw(pDC, 0, CPoint(rcImage.left + ax, rcImage.top + ay - 2), ILD_NORMAL);
  168. }
  169. }
  170. CString cs(pItem->cText), cs1;
  171. CSize sz;
  172. sz = pDC->GetTextExtent(cs);
  173. int ay1 = (rcText.Height() - sz.cy) / 2;
  174. rcText.top += ay1;
  175. rcText.left += 2;
  176. rcText.right -= 15;
  177. int tf = cs.Find('t');
  178. if (tf >= 0)
  179. {
  180. cs1 = cs.Right(cs.GetLength() - tf - 1);
  181. cs = cs.Left(tf);
  182. if (!bEnab)
  183. {
  184. if (!bSelect)
  185. {
  186. CRect rcText1(rcText);
  187. rcText1.InflateRect(-1,-1);
  188. pDC->SetTextColor(cr3dHilight);
  189. pDC->DrawText(cs, rcText1, DT_VCENTER|DT_LEFT);
  190. pDC->DrawText(cs1, rcText1, DT_VCENTER|DT_RIGHT);
  191. pDC->SetTextColor(crGrayText);
  192. pDC->DrawText(cs, rcText, DT_VCENTER|DT_LEFT);
  193. pDC->DrawText(cs1, rcText, DT_VCENTER|DT_RIGHT);
  194. }
  195. else
  196. {
  197. pDC->SetTextColor(crMenu);
  198. pDC->DrawText(cs, rcText, DT_VCENTER|DT_LEFT);
  199. pDC->DrawText(cs1, rcText, DT_VCENTER|DT_RIGHT);
  200. }
  201. }
  202. else
  203. {
  204. pDC->DrawText(cs, rcText, DT_VCENTER|DT_LEFT);
  205. pDC->DrawText(cs1, rcText, DT_VCENTER|DT_RIGHT);
  206. }
  207. }
  208. else 
  209. {
  210. if (!bEnab)
  211. {
  212. if (!bSelect)
  213. {
  214. CRect rcText1(rcText);
  215. rcText1.InflateRect(-1,-1);
  216. pDC->SetTextColor(cr3dHilight);
  217. pDC->DrawText(cs, rcText1, DT_VCENTER|DT_LEFT|DT_EXPANDTABS);
  218. pDC->SetTextColor(crGrayText);
  219. pDC->DrawText(cs, rcText, DT_VCENTER|DT_LEFT|DT_EXPANDTABS);
  220. }
  221. else
  222. {
  223. pDC->SetTextColor(crMenu);
  224. pDC->DrawText(cs, rcText, DT_VCENTER|DT_LEFT|DT_EXPANDTABS);
  225. }
  226. }
  227. else
  228. pDC->DrawText(cs, rcText, DT_VCENTER|DT_LEFT|DT_EXPANDTABS);
  229. }
  230. pDC->SetTextColor(ocr);
  231. pDC->SetBkMode(obk);
  232. }
  233. pDC->SelectObject(of);
  234. }
  235. }
  236. }
  237. void CGfxPopupMenu::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct)
  238. {
  239. // lpMeasureItemStruct->itemWidth = 200;
  240. // lpMeasureItemStruct->itemHeight = 25;
  241. bool res = false;
  242. if (lpMeasureItemStruct->CtlType == ODT_MENU)
  243. {
  244. UINT id = lpMeasureItemStruct->itemID;
  245. SpawnItem * pItem = (SpawnItem *) lpMeasureItemStruct->itemData;
  246. if (pItem)
  247. {
  248. if (pItem->iCmd == -3) // is a separator
  249. {
  250. lpMeasureItemStruct->itemWidth  = 10;
  251. lpMeasureItemStruct->itemHeight = 6;
  252. }
  253. else
  254. {
  255. CString cs(pItem->cText);
  256. if (!cs.IsEmpty())
  257. {
  258. CClientDC dc(AfxGetMainWnd());
  259. CFont * pft = CFont::FromHandle(hMenuFont ? hMenuFont : hGuiFont);
  260. CFont * of = dc.SelectObject(pft);
  261. CSize osz = dc.GetOutputTabbedTextExtent(cs,0,NULL);
  262. if (pItem->iCmd == -4)
  263. {
  264. CRect rci(0,0,0,0);
  265. dc.DrawText(cs, rci, DT_CALCRECT|DT_TOP|DT_VCENTER|DT_SINGLELINE);
  266. lpMeasureItemStruct->itemHeight = rci.Height();
  267. lpMeasureItemStruct->itemWidth = rci.Width();
  268. }
  269. else
  270. {
  271. lpMeasureItemStruct->itemHeight = szImage.cy + 5;
  272. if (osz.cy > (int) lpMeasureItemStruct->itemHeight) lpMeasureItemStruct->itemHeight = (int) osz.cy;
  273. lpMeasureItemStruct->itemWidth  = osz.cx + 2 + 15;
  274. lpMeasureItemStruct->itemWidth += lpMeasureItemStruct->itemHeight > (UINT) szImage.cx ? (UINT) lpMeasureItemStruct->itemHeight : (UINT) szImage.cx;
  275. }
  276. dc.SelectObject(of);
  277. }
  278. else
  279. {
  280. lpMeasureItemStruct->itemHeight = szImage.cy + 5;
  281. lpMeasureItemStruct->itemWidth  = 100;
  282. }
  283. }
  284. }
  285. }
  286. }
  287. bool CGfxPopupMenu::CreateBoldFont()
  288. {
  289. if (hMenuBoldFont) 
  290. ::DeleteObject((HGDIOBJ)hMenuBoldFont);
  291. LOGFONT lgFont;
  292. ::GetObject (hMenuFont ? hMenuFont : hGuiFont, sizeof (lgFont), &lgFont);
  293. lgFont.lfWeight = FW_BOLD;
  294. hMenuBoldFont = CreateFontIndirect (&lgFont);
  295. return !!hMenuBoldFont;
  296. }
  297. bool CGfxPopupMenu::AddToolBarResource(unsigned int resId)
  298. {
  299. // David 08/04/98 - start - put CMenuSpawn in DLL
  300. HINSTANCE hInst = AfxFindResourceHandle (MAKEINTRESOURCE(resId), RT_TOOLBAR);
  301. if (!hInst)
  302. return false;
  303. // David 08/04/98 - end - put CMenuSpawn in DLL
  304. HRSRC hRsrc = ::FindResource(/*AfxGetResourceHandle()*/hInst, MAKEINTRESOURCE(resId), RT_TOOLBAR);
  305. if (hRsrc == NULL) return false;
  306. HGLOBAL hGlb = ::LoadResource(/*AfxGetResourceHandle()*/hInst, hRsrc);
  307. if (hGlb == NULL) return false;
  308. ToolBarData* pTBData = (ToolBarData*) ::LockResource(hGlb);
  309. if (pTBData == NULL) return false;
  310. ASSERT(pTBData->wVersion == 1);
  311. CBitmap bmp;
  312. bmp.LoadBitmap(resId);
  313. int nBmpItems = ilList.Add(&bmp, RGB(192,192,192));
  314. bmp.DeleteObject();
  315. WORD* pItem = (WORD*)(pTBData+1);
  316. for(int i=0; i<pTBData->wItemCount; i++, pItem++)
  317. {
  318. if(*pItem != ID_SEPARATOR)
  319. AddImageItem(nBmpItems++, (WORD) *pItem);
  320. }
  321. // ** it seem that Windows doesn't free these resource (from Heitor Tome)
  322.     ::UnlockResource(hGlb);
  323.     ::FreeResource(hGlb);
  324. // **
  325. return true;
  326. }
  327. bool CGfxPopupMenu::LoadToolBarResource(unsigned int resId)
  328. {
  329. //David 08/04/98 - start - put CMenuSpawn in DLL
  330. HINSTANCE hInst = AfxFindResourceHandle (MAKEINTRESOURCE(resId), RT_TOOLBAR);
  331. if (!hInst)
  332. return false;
  333. //David 08/04/98 - end - put CMenuSpawn in DLL
  334. HRSRC hRsrc = ::FindResource(/*AfxGetResourceHandle()*/hInst, MAKEINTRESOURCE(resId), RT_TOOLBAR);
  335. if (hRsrc == NULL) return false;
  336. HGLOBAL hGlb = ::LoadResource(/*AfxGetResourceHandle()*/hInst, hRsrc);
  337. if (hGlb == NULL) return false;
  338. ToolBarData* pTBData = (ToolBarData*) ::LockResource(hGlb);
  339. if (pTBData == NULL) return false;
  340. ASSERT(pTBData->wVersion == 1);
  341. szImage.cx = (int) pTBData->wWidth;
  342. szImage.cy = (int) pTBData->wHeight;
  343. if (ilList.Create(szImage.cx, szImage.cy, ILC_COLOR4|ILC_MASK, pTBData->wItemCount, 0) == false)
  344. return false;
  345. ilList.SetBkColor(cr3dFace);
  346. CBitmap bmp;
  347. bmp.LoadBitmap(resId);
  348. ilList.Add(&bmp, RGB(192,192,192));
  349. bmp.DeleteObject();
  350. WORD* pItem = (WORD*)(pTBData+1);
  351. int nBmpItems = 0;
  352. for(int i=0; i<pTBData->wItemCount; i++, pItem++)
  353. {
  354. if(*pItem != ID_SEPARATOR)
  355. AddImageItem(nBmpItems++, (WORD) *pItem);
  356. }
  357. // ** it seem that Windows doesn't free these resource (from Heitor Tome)
  358.     ::UnlockResource(hGlb);
  359.     ::FreeResource(hGlb);
  360. // **
  361. return true;
  362. }
  363. void CGfxPopupMenu::AddImageItem(const int idx, WORD cmd)
  364. {
  365. if (iImageItem == 0)
  366. pImageItem = (ImageItem *) GlobalAlloc(GPTR, sizeof(ImageItem));
  367. else
  368. pImageItem = (ImageItem *) GlobalReAlloc((HGLOBAL) pImageItem, sizeof(ImageItem) * (iImageItem + 1), GMEM_MOVEABLE|GMEM_ZEROINIT);
  369. ASSERT(pImageItem);
  370. pImageItem[iImageItem].iCmd = (int) cmd;
  371. pImageItem[iImageItem].iImageIdx = idx;
  372. iImageItem ++;
  373. }
  374. void CGfxPopupMenu::RemapMenu(CMenu * pMenu)
  375. {
  376. static int iRecurse = 0;
  377. iRecurse ++;
  378. ASSERT(pMenu);
  379. int nItem = pMenu->GetMenuItemCount();
  380. while ((--nItem)>=0)
  381. {
  382. UINT itemId = pMenu->GetMenuItemID(nItem);
  383. if (itemId == (UINT) -1)
  384. {
  385. CMenu *pops = pMenu->GetSubMenu(nItem);
  386. if (pops) RemapMenu(pops);
  387. if (iRecurse > 0)
  388. {
  389. CString cs;
  390. pMenu->GetMenuString(nItem, cs, MF_BYPOSITION);
  391. if (cs != "")
  392. {
  393. SpawnItem * sp = AddSpawnItem(cs, (iRecurse == 1) ? -4 : -2);
  394. pMenu->ModifyMenu(nItem,MF_BYPOSITION|MF_OWNERDRAW, (UINT) -1, (LPCTSTR)sp);
  395. }
  396. }
  397. }
  398. else
  399. {
  400. if (itemId != 0)
  401. {
  402. UINT oldState = pMenu->GetMenuState(nItem,MF_BYPOSITION);
  403. if (!(oldState&MF_OWNERDRAW) && !(oldState&MF_BITMAP))
  404. {
  405. ASSERT(oldState != (UINT)-1);
  406. CString cs;
  407. pMenu->GetMenuString(nItem, cs, MF_BYPOSITION);
  408. SpawnItem * sp = AddSpawnItem(cs, itemId);
  409. pMenu->ModifyMenu(nItem,MF_BYPOSITION|MF_OWNERDRAW|oldState, (LPARAM)itemId, (LPCTSTR)sp);
  410. }
  411. }
  412. else
  413. {
  414. UINT oldState = pMenu->GetMenuState(nItem,MF_BYPOSITION);
  415. if (!(oldState&MF_OWNERDRAW) && !(oldState&MF_BITMAP))
  416. {
  417. ASSERT(oldState != (UINT)-1);
  418. SpawnItem * sp = AddSpawnItem("--", -3);
  419. pMenu->ModifyMenu(nItem,MF_BYPOSITION|MF_OWNERDRAW|oldState, (LPARAM)itemId, (LPCTSTR)sp);
  420. }
  421. }
  422. }
  423. }
  424. iRecurse --;
  425. }
  426. CGfxPopupMenu::SpawnItem * CGfxPopupMenu::AddSpawnItem(const char * txt, const int cmd)
  427. {
  428. if (iSpawnItem == 0)
  429. pSpawnItem = (SpawnItem **) GlobalAlloc(GPTR, sizeof(SpawnItem));
  430. else
  431. pSpawnItem = (SpawnItem **) GlobalReAlloc((HGLOBAL) pSpawnItem, sizeof(SpawnItem) * (iSpawnItem + 1), GMEM_MOVEABLE|GMEM_ZEROINIT);
  432. ASSERT(pSpawnItem);
  433. SpawnItem * p = new SpawnItem;
  434. ASSERT(p);
  435. pSpawnItem[iSpawnItem] = p;
  436. lstrcpy(p->cText, txt);
  437. p->iCmd = cmd;
  438. if (cmd >= 0) p->iImageIdx = FindImageItem(cmd);
  439. else p->iImageIdx = cmd;
  440. iSpawnItem ++;
  441. return p;
  442. }
  443. int CGfxPopupMenu::FindImageItem(const int cmd)
  444. {
  445. for (int t = 0; t < iImageItem; t++)
  446. if (pImageItem[t].iCmd == cmd) return pImageItem[t].iImageIdx;
  447. return -1;
  448. }
  449. void CGfxPopupMenu::EnableMenuItems(CMenu * pMenu, CWnd * pParent)
  450. {
  451. ASSERT(pMenu);
  452. ASSERT(pParent);
  453. int nItem = pMenu->GetMenuItemCount();
  454. CCmdUI state;
  455. state.m_pMenu = pMenu;
  456. state.m_nIndex = nItem-1;
  457. state.m_nIndexMax = nItem;
  458. while ((--nItem)>=0)
  459. {
  460. UINT itemId = pMenu->GetMenuItemID(nItem);
  461. if (itemId == (UINT) -1)
  462. {
  463. CMenu *pops = pMenu->GetSubMenu(nItem);
  464. if (pops) EnableMenuItems(pops, pParent);
  465. }
  466. else
  467. {
  468. if (itemId != 0)
  469. {
  470. state.m_nID = itemId;
  471. pParent->OnCmdMsg(itemId, CN_UPDATE_COMMAND_UI, &state, NULL);
  472. state.DoUpdate(pParent, true);
  473. }
  474. }
  475. state.m_nIndex = nItem-1;
  476. }
  477. }