📄 popupmenu.cpp
字号:
// PopupMenu.cpp: implementation of the CPopupMenu class.
//
//////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "PopupMenu.h"
#ifdef _DEBUG
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#define new DEBUG_NEW
#endif
#ifndef OBM_CHECK
#define OBM_CHECK 32760 // from winuser.h
#endif
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
CPopupMenu::CPopupMenu()
{
//取得各种状态下的颜色
crMenuText = GetSysColor(COLOR_MENUTEXT);
crMenuTextSel = GetSysColor(COLOR_HIGHLIGHTTEXT);
cr3dFace = GetSysColor(COLOR_3DFACE);
crMenu = GetSysColor(COLOR_MENU);
crHighlight = GetSysColor(COLOR_HIGHLIGHT);
cr3dHilight = GetSysColor(COLOR_3DHILIGHT);
cr3dShadow = GetSysColor(COLOR_3DSHADOW);
crGrayText = GetSysColor(COLOR_GRAYTEXT);
m_clrBtnFace = GetSysColor(COLOR_BTNFACE);
m_clrBtnHilight = GetSysColor(COLOR_BTNHILIGHT);
m_clrBtnShadow = GetSysColor(COLOR_BTNSHADOW);
iSpawnItem = 0;
pSpawnItem = NULL;
iImageItem = 0;
pImageItem = NULL;
//图像尺寸
szImage = CSize(20, 20);
hMenuFont = NULL;
NONCLIENTMETRICS ncm;
memset(&ncm, 0, sizeof(ncm));
ncm.cbSize = sizeof(ncm);
::SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, static_cast<PVOID>(&ncm), 0);
//图像状态字体
hGuiFont = ::CreateFontIndirect(&ncm.lfMenuFont);
// David 08/04/98 - start - bold font handling
hMenuBoldFont = NULL;
CreateBoldFont();
// David 08/04/98 - end - bold font handling
m_bFirstRun = TRUE;
}
CPopupMenu::~CPopupMenu()
{
if (iSpawnItem > 0)
{
for (int t = 0; t < iSpawnItem; t++)
{
if (pSpawnItem[t])
delete pSpawnItem[t];
}
GlobalFree(static_cast<HGLOBAL>(pSpawnItem));
}
if (iImageItem > 0)
{
GlobalFree(static_cast<HGLOBAL>(pImageItem));
}
if (hMenuFont)
::DeleteObject(static_cast<HGDIOBJ>(hMenuFont));
if (hMenuBoldFont)
::DeleteObject(static_cast<HGDIOBJ>(hMenuBoldFont));
}
void CPopupMenu::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
{
if (lpDrawItemStruct->CtlType == ODT_MENU)
{
//取回菜单的各种状态
UINT state = lpDrawItemStruct->itemState;
BOOL bEnab = !(state & ODS_DISABLED);
BOOL bSelect = (state & ODS_SELECTED) ? TRUE : FALSE;
BOOL bChecked = (state & ODS_CHECKED) ? TRUE : FALSE;
BOOL bBold = (state & ODS_DEFAULT) ? TRUE : FALSE;
//取回菜单相应数据
SpawnItem* pItem = reinterpret_cast<SpawnItem*>(lpDrawItemStruct->itemData);
if (pItem)
{
CDC* pDC = CDC::FromHandle(lpDrawItemStruct->hDC);
CFont* pft;
//是否黑体显示
if (!bBold)
pft = CFont::FromHandle(static_cast<HFONT>(hMenuFont) ? hMenuFont : hGuiFont);
else
pft = CFont::FromHandle(static_cast<HFONT>(hMenuBoldFont) ? hMenuBoldFont : hGuiFont);
CFont* of = pDC->SelectObject(pft);
CRect rc(lpDrawItemStruct->rcItem);
CRect rcImage(rc), rcText(rc);
//调整图像显示区域
rcImage.right = rcImage.left + rc.Height();
rcImage.bottom = rc.bottom;
if (pItem->iCmd == -3) // 分隔条
{
CPen pnDk(PS_SOLID, 1, cr3dShadow);
CPen pnLt(PS_SOLID, 1, cr3dHilight);
CPen * opn = pDC->SelectObject(&pnDk);
pDC->MoveTo(rc.left + 2, rc.top + 2);
pDC->LineTo(rc.right - 2, rc.top + 2);
pDC->SelectObject(&pnLt);
pDC->MoveTo(rc.left + 2, rc.top + 3);
pDC->LineTo(rc.right - 2, rc.top + 3);
pDC->SelectObject(opn);
}
else if (pItem->iCmd == -4) // 标题条,这一段代码没有执行到
{
CString cs(pItem->cText), cs1;
CRect rcBdr(rcText);
if (bSelect && bEnab)
{
rcText.top ++;
rcText.left += 2;
}
pDC->FillSolidRect(rcText, crMenu);
pDC->DrawText(cs, rcText, DT_VCENTER | DT_CENTER | DT_SINGLELINE);
if (bSelect && bEnab)
pDC->Draw3dRect(rcBdr, cr3dShadow, cr3dHilight);
}
else //正常的菜单项
{
rcText.left += rcImage.right + 1;
int obk = pDC->SetBkMode(TRANSPARENT); //输出透明
COLORREF ocr;
if (bSelect) //选择状态
{
if (pItem->iImageIdx >= 0 || (state & ODS_CHECKED)) //CHECKED
pDC->FillSolidRect(rcText, crHighlight);
else
pDC->FillSolidRect(rc, crHighlight); //UNCHECKED
ocr = pDC->SetTextColor(crMenuTextSel);
}
else //非选择状态
{
if (pItem->iImageIdx >= 0 || (state & ODS_CHECKED)) //CHECKED
pDC->FillSolidRect(rcText, crMenu);
else
pDC->FillSolidRect(rc/*rcText*/, crMenu); //UNCHECKED
ocr = pDC->SetTextColor(crMenuText);
}
if (pItem->iImageIdx >= 0) //有图像
{
int ay = (rcImage.Height() - szImage.cy) / 2;
int ax = (rcImage.Width() - szImage.cx) / 2;
if (bSelect && bEnab)
pDC->Draw3dRect(rcImage, cr3dHilight, cr3dShadow); //选择且允许状态
else
pDC->Draw3dRect(rcImage, crMenu, crMenu); //非选择或禁止状态
if (bEnab) //允许状态
{
ilList.Draw(pDC, pItem->iImageIdx, CPoint(rcImage.left + ax,
rcImage.top +ay), ILD_NORMAL);
}
else //禁止状态,注意画禁止状态图像的代码
{
HICON hIcon = ilList.ExtractIcon(pItem->iImageIdx);
pDC->DrawState(CPoint(rcImage.left + ax, rcImage.top + ay), szImage,
static_cast<HICON>(hIcon), DST_ICON | DSS_DISABLED,
static_cast<CBrush*>(NULL));
}
}
else //没有图像
{
if (bChecked) //CHECKED
{
CRect rcTemp = rcImage;
rcTemp.DeflateRect(2, 2);
DrawCheckmark(*pDC, rcTemp, bSelect);
}
if (bSelect || lpDrawItemStruct->itemAction == ODA_SELECT) //选择
{
COLORREF colorBG = GetSysColor(bSelect ? COLOR_HIGHLIGHT : COLOR_MENU);
// selected or selection state changed: paint text background
CRect rcBG(lpDrawItemStruct->rcItem); // the whole rectangle
rcBG.left += rcImage.Width(); // do not paint over it!
CBrush brush(colorBG);
CBrush* pOldBrush = pDC->SelectObject(&brush);
pDC->PatBlt(rcBG.left, rcBG.top, rcBG.Width(), rcBG.Height(), PATCOPY);
pDC->SelectObject(pOldBrush);
}
}
//输出菜单文本
CString cs(pItem->cText), cs1;
CSize sz;
sz = pDC->GetTextExtent(cs);
int ay1 = (rcText.Height() - sz.cy) / 2;
rcText.top += ay1;
rcText.left += 2;
rcText.right -= 15;
//找到快捷键
int tf = cs.Find('\t');
if (tf >= 0) //有快捷键
{
cs1 = cs.Right(cs.GetLength() - tf - 1);
cs = cs.Left(tf);
//禁止状态
if (!bEnab)
{
//非选择状态
if (!bSelect)
{
CRect rcText1(rcText);
rcText1.InflateRect(-1, - 1);
pDC->SetTextColor(cr3dHilight);
pDC->DrawText(cs, rcText1, DT_VCENTER | DT_LEFT);
pDC->DrawText(cs1, rcText1, DT_VCENTER | DT_RIGHT);
//有阴影,画两次
pDC->SetTextColor(crGrayText);
pDC->DrawText(cs, rcText, DT_VCENTER | DT_LEFT);
pDC->DrawText(cs1, rcText, DT_VCENTER | DT_RIGHT);
}
else //选择状态
{
pDC->SetTextColor(crMenu);
pDC->DrawText(cs, rcText, DT_VCENTER | DT_LEFT);
pDC->DrawText(cs1, rcText, DT_VCENTER | DT_RIGHT);
}
}
else //允许状态
{
pDC->DrawText(cs, rcText, DT_VCENTER | DT_LEFT);
pDC->DrawText(cs1, rcText, DT_VCENTER | DT_RIGHT);
}
}
else //没有快捷键
{
if (!bEnab) //禁止状态
{
if (!bSelect) //非选择状态
{
CRect rcText1(rcText);
rcText1.InflateRect(-1, - 1);
pDC->SetTextColor(cr3dHilight);
pDC->DrawText(cs, rcText1, DT_VCENTER | DT_LEFT | DT_EXPANDTABS);
pDC->SetTextColor(crGrayText);
pDC->DrawText(cs, rcText, DT_VCENTER | DT_LEFT | DT_EXPANDTABS);
}
else //选择状态
{
pDC->SetTextColor(crMenu);
pDC->DrawText(cs, rcText, DT_VCENTER | DT_LEFT | DT_EXPANDTABS);
}
}
else //允许状态
pDC->DrawText(cs, rcText, DT_VCENTER | DT_LEFT | DT_EXPANDTABS);
}
pDC->SetTextColor(ocr);
pDC->SetBkMode(obk);
}
pDC->SelectObject(of);
}
}
}
void CPopupMenu::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct)
{
// lpMeasureItemStruct->itemWidth = 200;
// lpMeasureItemStruct->itemHeight = 25;
if (lpMeasureItemStruct->CtlType == ODT_MENU)
{
//取得菜单数据
SpawnItem* pItem = reinterpret_cast<SpawnItem*>(lpMeasureItemStruct->itemData);
if (pItem)
{
//分隔条
if (pItem->iCmd == -3) // is a separator
{
lpMeasureItemStruct->itemWidth = 10;
lpMeasureItemStruct->itemHeight = 6;
}
else
{
CString cs(pItem->cText);
if (!cs.IsEmpty())
{
CClientDC dc(AfxGetMainWnd());
CFont* pft = CFont::FromHandle(hMenuFont ? hMenuFont : hGuiFont);
CFont* of = dc.SelectObject(pft);
CSize osz = dc.GetOutputTabbedTextExtent(cs, 0, NULL);
if (pItem->iCmd == -4) //标题条
{
CRect rci(0, 0, 0, 0);
dc.DrawText(cs, rci, DT_CALCRECT | DT_TOP | DT_VCENTER | DT_SINGLELINE);
lpMeasureItemStruct->itemHeight = rci.Height();
lpMeasureItemStruct->itemWidth = rci.Width();
}
else
{
//高度为图像高度加5
lpMeasureItemStruct->itemHeight = szImage.cy + 5;
if (osz.cy > static_cast<int>(lpMeasureItemStruct->itemHeight))
lpMeasureItemStruct->itemHeight = static_cast<int>(osz.cy);
lpMeasureItemStruct->itemWidth = osz.cx + 2 + 15;
lpMeasureItemStruct->itemWidth += lpMeasureItemStruct->itemHeight > static_cast<UINT>(szImage.cx)
? static_cast<UINT>(lpMeasureItemStruct->itemHeight) : static_cast<UINT>(szImage.cx);
}
dc.SelectObject(of);
}
else
{
lpMeasureItemStruct->itemHeight = szImage.cy + 5;
lpMeasureItemStruct->itemWidth = 100;
}
}
}
}
}
BOOL CPopupMenu::CreateBoldFont()
{
if (hMenuBoldFont)
::DeleteObject(static_cast<HGDIOBJ>(hMenuBoldFont));
LOGFONT lgFont;
::GetObject (hMenuFont ? hMenuFont : hGuiFont, sizeof(lgFont), &lgFont);
lgFont.lfWeight = FW_BOLD;
hMenuBoldFont = CreateFontIndirect(&lgFont);
return !!hMenuBoldFont;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -