📄 popupmenu.cpp
字号:
}
BOOL CPopupMenu::AddToolBarResource(UINT resId)
{
// David 08/04/98 - start - put CMenuSpawn in DLL
HINSTANCE hInst = AfxFindResourceHandle(MAKEINTRESOURCE(resId), RT_TOOLBAR);
if (!hInst)
return FALSE;
// David 08/04/98 - end - put CMenuSpawn in DLL
HRSRC hRsrc = ::FindResource(/*AfxGetResourceHandle()*/hInst, MAKEINTRESOURCE(resId), RT_TOOLBAR);
if (hRsrc == NULL)
return FALSE;
HGLOBAL hGlb = ::LoadResource(/*AfxGetResourceHandle()*/hInst, hRsrc);
if (hGlb == NULL)
return FALSE;
ToolBarData* pTBData = static_cast<ToolBarData*>(::LockResource(hGlb));
if (pTBData == NULL)
return FALSE;
ASSERT(pTBData->wVersion == 1);
CBitmap bmp;
bmp.LoadBitmap(resId);
int nBmpItems = ilList.Add(&bmp, RGB(192, 192, 192));
bmp.DeleteObject();
WORD* pItem = reinterpret_cast<WORD*>(pTBData + 1);
for (int i = 0; i < pTBData->wItemCount; i++, pItem++)
{
if (*pItem != ID_SEPARATOR)
AddImageItem(nBmpItems++, static_cast<WORD>(*pItem));
}
// ** it seem that Windows doesn't free these resource (from Heitor Tome)
::UnlockResource(hGlb);
::FreeResource(hGlb);
// **
return TRUE;
}
BOOL CPopupMenu::LoadToolBarResource(UINT resId)
{
// David 08/04/98 - start - put CMenuSpawn in DLL
HINSTANCE hInst = AfxFindResourceHandle(MAKEINTRESOURCE(resId), RT_TOOLBAR);
if (!hInst)
return FALSE;
// David 08/04/98 - end - put CMenuSpawn in DLL
HRSRC hRsrc = ::FindResource(/*AfxGetResourceHandle()*/hInst, MAKEINTRESOURCE(resId), RT_TOOLBAR);
if (hRsrc == NULL)
return FALSE;
HGLOBAL hGlb = ::LoadResource(/*AfxGetResourceHandle()*/hInst, hRsrc);
if (hGlb == NULL)
return FALSE;
//找到工具栏资源
ToolBarData* pTBData = static_cast<ToolBarData*>(::LockResource(hGlb));
if (pTBData == NULL)
return FALSE;
ASSERT(pTBData->wVersion == 1);
//图像尺寸
szImage.cx = static_cast<int>(pTBData->wWidth);
szImage.cy = static_cast<int>(pTBData->wHeight);
//创建一个CImageList对象
if (ilList.Create(szImage.cx, szImage.cy, ILC_COLOR4 | ILC_MASK, pTBData->wItemCount, 0) == FALSE)
return FALSE;
ilList.SetBkColor(cr3dFace);
CBitmap bmp;
bmp.LoadBitmap(resId);
//将工具栏位图装入
ilList.Add(&bmp, RGB(192, 192, 192));
bmp.DeleteObject();
//收集工具栏中命令ID不为分隔符的ID
WORD* pItem = reinterpret_cast<WORD*>(pTBData + 1);
int nBmpItems = 0;
for (int i = 0; i < pTBData->wItemCount; i++, pItem++)
{
if (*pItem != ID_SEPARATOR)
AddImageItem(nBmpItems++, static_cast<WORD>(*pItem));
}
// ** it seem that Windows doesn't free these resource (from Heitor Tome)
::UnlockResource(hGlb);
::FreeResource(hGlb);
// **
return TRUE;
}
void CPopupMenu::AddImageItem(const int idx, WORD cmd)
{
if (iImageItem == 0)
pImageItem = static_cast<ImageItem*>(GlobalAlloc(GPTR, sizeof(ImageItem)));
else
pImageItem = static_cast<ImageItem*>(GlobalReAlloc(static_cast<HGLOBAL>(pImageItem),
sizeof(ImageItem) * (iImageItem + 1), GMEM_MOVEABLE | GMEM_ZEROINIT));
ASSERT(pImageItem);
pImageItem[iImageItem].iCmd = static_cast<int>(cmd); //命令ID
pImageItem[iImageItem].iImageIdx = idx; //图像索引
iImageItem ++;
}
//重新设置每个菜单项的数据
void CPopupMenu::RemapMenu(CMenu* pMenu)
{
static int iRecurse = 0;
iRecurse ++;
ASSERT(pMenu);
int nItem = pMenu->GetMenuItemCount();
while ((--nItem) >= 0)
{
UINT itemId = pMenu->GetMenuItemID(nItem);
if (itemId == static_cast<UINT>(-1))
{
//弹出式菜单
CMenu* pops = pMenu->GetSubMenu(nItem);
if (pops)
RemapMenu(pops);
if (iRecurse > 0)
{
CString cs;
pMenu->GetMenuString(nItem, cs, MF_BYPOSITION);
if (cs != "")
{
SpawnItem* sp = AddSpawnItem(cs, (iRecurse == 1) ? -4 : -2);
pMenu->ModifyMenu(nItem, MF_BYPOSITION | MF_OWNERDRAW,
static_cast<UINT>(-1), reinterpret_cast<LPCTSTR>(sp));
}
}
}
else
{
if (itemId != 0)
{
//菜单项
UINT oldState = pMenu->GetMenuState(nItem, MF_BYPOSITION);
if (!(oldState & MF_OWNERDRAW) && !(oldState & MF_BITMAP))
{
ASSERT(oldState != (UINT) - 1);
CString cs;
pMenu->GetMenuString(nItem, cs, MF_BYPOSITION);
SpawnItem* sp = AddSpawnItem(cs, itemId);
pMenu->ModifyMenu(nItem, MF_BYPOSITION | MF_OWNERDRAW | oldState,
static_cast<LPARAM>(itemId), reinterpret_cast<LPCTSTR>(sp));
}
}
else
{
//分隔条
UINT oldState = pMenu->GetMenuState(nItem, MF_BYPOSITION);
if (!(oldState & MF_OWNERDRAW) && !(oldState & MF_BITMAP))
{
ASSERT(oldState != static_cast<UINT>(-1));
SpawnItem* sp = AddSpawnItem(_T("--"), -3);
pMenu->ModifyMenu(nItem, MF_BYPOSITION | MF_OWNERDRAW | oldState,
static_cast<LPARAM>(itemId), reinterpret_cast<LPCTSTR>(sp));
}
}
}
}
iRecurse --;
}
//将一个菜单数据加入数组
CPopupMenu::SpawnItem * CPopupMenu::AddSpawnItem(const TCHAR* txt, const int cmd)
{
if (iSpawnItem == 0)
pSpawnItem = static_cast<SpawnItem**>(GlobalAlloc(GPTR, sizeof(SpawnItem)));
else
pSpawnItem = static_cast<SpawnItem**>(GlobalReAlloc(static_cast<HGLOBAL>(pSpawnItem),
sizeof(SpawnItem) * (iSpawnItem + 1), GMEM_MOVEABLE | GMEM_ZEROINIT));
ASSERT(pSpawnItem);
SpawnItem* p = new SpawnItem;
ASSERT(p);
pSpawnItem[iSpawnItem] = p;
lstrcpy(p->cText, txt);
p->iCmd = cmd;
if (cmd >= 0)
p->iImageIdx = FindImageItem(cmd);
else p->iImageIdx = cmd;
iSpawnItem++;
return p;
}
//找到和命令ID cmd对应的图像索引
int CPopupMenu::FindImageItem(const int cmd)
{
for (int t = 0; t < iImageItem; t++)
{
if (pImageItem[t].iCmd == cmd)
return pImageItem[t].iImageIdx;
}
return -1;
}
//在菜单引出以前触发每个菜单的ON_XXXX_UPDATE_UI,更新菜单状态
void CPopupMenu::EnableMenuItems(CMenu* pMenu, CWnd* pParent)
{
ASSERT(pMenu);
ASSERT(pParent);
int nItem = pMenu->GetMenuItemCount();
CCmdUI state;
state.m_pMenu = pMenu;
state.m_nIndex = nItem - 1;
state.m_nIndexMax = nItem;
while ((--nItem) >= 0)
{
UINT itemId = pMenu->GetMenuItemID(nItem);
if (itemId == static_cast<UINT>(-1))//弹出式菜单
{
CMenu* pops = pMenu->GetSubMenu(nItem);
if (pops)
EnableMenuItems(pops, pParent);
}
else
{
if (itemId != 0)
{
state.m_nID = itemId;
pParent->OnCmdMsg(itemId, CN_UPDATE_COMMAND_UI, &state, NULL);
state.DoUpdate(pParent, TRUE);
}
}
state.m_nIndex = nItem - 1;
}
}
//画Check
BOOL CPopupMenu::DrawCheckmark(CDC& dc, const CRect& rc, BOOL bSelected)
{
// Use Windows standard
HBITMAP hbmCheck = ::LoadBitmap(NULL, reinterpret_cast<LPCTSTR>(OBM_CHECK));
// Center bitmap in distination rectangle
BITMAP bm;
::GetObject(hbmCheck, sizeof(bm), &bm);
int cx = bm.bmWidth;
int cy = bm.bmHeight;
CRect rcDest = rc;
CPoint ptSrc = -CPoint((rc.Width() - cx)/2, (rc.Height() - cy)/2);
// Select checkmark into memory DC
CDC memdc;
memdc.CreateCompatibleDC(&dc);
HBITMAP hOldBM = reinterpret_cast<HBITMAP>(::SelectObject(memdc, hbmCheck));
// Set background color based on selected state
COLORREF colorOld = dc.SetBkColor(GetSysColor(bSelected ? COLOR_MENU : COLOR_3DLIGHT));
dc.BitBlt(rcDest.left, rcDest.top, rcDest.Width(), rcDest.Height(),
&memdc, ptSrc.x, ptSrc.y, SRCCOPY);
dc.SetBkColor(colorOld);
::SelectObject(memdc, hOldBM); // restore
// draw pushed-in hilight.
dc.DrawEdge(&rcDest, BDR_SUNKENOUTER, BF_RECT);
return TRUE;
}
void CPopupMenu::Load(UINT nID)
{
CMenu tempMenu;
tempMenu.LoadMenu(nID);
ConvertMenu(tempMenu.GetSubMenu(0),this);
if(m_bFirstRun)
{
int nItem = GetMenuItemCount();
for(int i= 0; i< nItem; i++)
{
UINT iState = 0;
iState = GetMenuState(i,MF_BYPOSITION);
m_arrayItemState.Add(iState);
}
}
m_bFirstRun = FALSE;
}
void CPopupMenu::ConvertMenu(CMenu *pSourceMenu, CMenu *pDestMenu)
{
int nItem = pSourceMenu->GetMenuItemCount();
UINT nMenuID;
CString strMenuText;
for(int i=0; i<nItem; i++)
{
nMenuID = pSourceMenu->GetMenuItemID(i);
pSourceMenu->GetMenuString(i,strMenuText,
MF_BYPOSITION);
UINT iState = pSourceMenu->GetMenuState(i,MF_BYPOSITION);
if(nMenuID == -3)
pDestMenu->AppendMenu(MF_SEPARATOR);
else if(nMenuID == -1) //不支持嵌套菜单
{
// CMenu* pMenu = pSourceMenu->GetSubMenu(i);
// CMenu* pMenu1 = new CMenu;
// pMenu1->CreatePopupMenu();
// pDestMenu->AppendMenu(MF_POPUP|MF_OWNERDRAW,(UINT)(HMENU)(pMenu1->m_hMenu),strMenuText);
// ConvertMenu(pMenu,pMenu1);
}
else
pDestMenu->AppendMenu(MF_STRING|iState,nMenuID,strMenuText);
}
}
void CPopupMenu::Enable(UINT nID, BOOL bEnable)
{
int nItem = GetMenuItemCount();
for(int i=0; i<nItem; i++)
{
if(GetMenuItemID(i) == nID)
{
UINT iState = m_arrayItemState.GetAt(i);
if(!bEnable)
iState |= MF_GRAYED;
else
iState |= MF_ENABLED;
m_arrayItemState.SetAt(i,1);
}
}
}
void CPopupMenu::Init()
{
for(int i =0; i<m_arrayItemState.GetSize(); i++)
{
int iState = m_arrayItemState.GetAt(i);
EnableMenuItem(i,MF_BYPOSITION|iState);
}
}
void CPopupMenu::Display(CPoint point, CWnd *pParentWnd)
{
Init();
TrackPopupMenu(TPM_LEFTALIGN,point.x,point.y, pParentWnd);
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -