自绘制带图标的菜单项

简介:
const int MAX_MENUCOUNT = 60,ITEMHEIGHT = 26,ITEMWIDTH= 120; //菜单项数量


/*************************************
CMenuItemInfo结构用于记录菜单项信息
*************************************/
struct CMenuItemInfo
{
    CString m_ItemText;//菜单项文本
    int m_IconIndex;//菜单项索引
    int m_ItemID;//菜单标记-2顶层菜单,-1弹出式菜单,0分隔条,其他普通菜单
};


class CIconMenu : public CMenu  
{
public:
    CMenuItemInfo m_ItemLists[MAX_MENUCOUNT]; //菜单项信息
    int m_index; //临时索引
    int m_iconindex;
    CImageList m_imagelist;//存储菜单项图标

    CIconMenu();
    BOOL ChangeMenuItem(CMenu* m_menu,BOOL m_Toped = FALSE);
    BOOL  AttatchMenu(HMENU m_hmenu);
    void DrawItemText(CDC* m_pdc,LPSTR str,CRect m_rect);
    void DrawTopMenu(CDC* m_pdc,CRect m_rect,BOOL m_selected = FALSE); //绘制顶层菜单
    void DrawSeparater(CDC* m_pdc,CRect m_rect);//绘制分隔条
    void DrawComMenu(CDC* m_pdc,CRect m_rect,COLORREF m_fromcolor,COLORREF m_tocolor, BOOL m_selected = FALSE);
    void DrawMenuIcon(CDC* m_pdc,CRect m_rect,int m_icon );
    
    //override memu's viutual method
    virtual ~CIconMenu();
    virtual void MeasureItem( LPMEASUREITEMSTRUCT lpStruct );//设置菜单项大小
    virtual void DrawItem( LPDRAWITEMSTRUCT lpStruct );//重绘菜单项

};



CIconMenu::CIconMenu()
{
    m_index= 0;
    m_iconindex= 0;
    //创建图像列表
    m_imagelist.Create(16,16,ILC_COLOR24|ILC_MASK,0,0);

    //添加图标
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON1));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON2));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON3));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON4));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON5));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON6));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON7));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON8));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON9));
    m_imagelist.Add(AfxGetApp()->LoadIcon(IDI_ICON10));
}

CIconMenu::~CIconMenu()
{
    m_imagelist.Detach();
}

BOOL  CIconMenu::AttatchMenu(HMENU m_hmenu)
{
    this->Attach(m_hmenu);
    return TRUE;
}

BOOL CIconMenu::ChangeMenuItem(CMenu* m_menu,BOOL m_Toped)
{
    if (m_menu != NULL)
    {
        int m_itemcount = m_menu->GetMenuItemCount();
        for (int i=0;i<m_itemcount;i++)
        {
            m_menu->GetMenuString(i,m_ItemLists[m_index].m_ItemText,MF_BYPOSITION);
                int m_itemID = m_menu->GetMenuItemID(i);
            if (m_itemID==-1 && m_Toped)
            {
                m_itemID = -2;//顶层菜单
            };
            m_ItemLists[m_index].m_ItemID = m_itemID;
            if (m_itemID>0)
            {
                m_ItemLists[m_index].m_IconIndex= m_iconindex;
                m_iconindex+=1;
            }
            m_menu->ModifyMenu(i,MF_OWNERDRAW|MF_BYPOSITION |MF_STRING,m_ItemLists[m_index].m_ItemID,(LPSTR)&(m_ItemLists[m_index]));
            
            m_index+=1;
            CMenu* m_subMenu = m_menu->GetSubMenu(i);

            if (m_subMenu)
            {
                ChangeMenuItem(m_subMenu);
            }
        }
    }
    return TRUE    ;
}

void CIconMenu::MeasureItem( LPMEASUREITEMSTRUCT lpStruct )
{
    if (lpStruct->CtlType==ODT_MENU)
    {
        lpStruct->itemHeight = ITEMHEIGHT;
        lpStruct->itemWidth = ITEMWIDTH;
        CMenuItemInfo* m_iteminfo;
        m_iteminfo = (CMenuItemInfo*)lpStruct->itemData;
        lpStruct->itemWidth = ((CMenuItemInfo*)lpStruct->itemData)->m_ItemText.GetLength()*10;
        switch(m_iteminfo->m_ItemID)
        {
            case 0: //分隔条
                {
                    lpStruct->itemHeight = 1;
                    break;
                }    
        }
    }
}

void CIconMenu::DrawItem( LPDRAWITEMSTRUCT lpStruct )
{
    if (lpStruct->CtlType==ODT_MENU)
    {
        if(lpStruct->itemData == NULL)    return;
        unsigned int m_state = lpStruct->itemState;
        CDC* m_dc = CDC::FromHandle(lpStruct->hDC);
        //m_dc.Attach(lpStruct->hDC);

        CString str =  ((CMenuItemInfo*)(lpStruct->itemData))->m_ItemText;
        LPSTR m_str = str.GetBuffer(str.GetLength());

        int m_itemID = ((CMenuItemInfo*)(lpStruct->itemData))->m_ItemID;
        int m_itemicon = ((CMenuItemInfo*)(lpStruct->itemData))->m_IconIndex;
        CRect m_rect = lpStruct->rcItem;

        m_dc->SetBkMode(TRANSPARENT);

        switch(m_itemID)
        {
        case -2:
            {

                DrawTopMenu(m_dc,m_rect,(m_state&ODS_SELECTED)||(m_state&0x0040)); //0x0040 ==ODS_HOTLIGHT

                DrawItemText(m_dc,m_str,m_rect);
                break;
            }
        case -1:
            {    
                DrawItemText(m_dc,m_str,m_rect);
                break;
            }
        case 0:
            {
                DrawSeparater(m_dc,m_rect);
                break;
            }
        default:
            {        
                DrawComMenu(m_dc,m_rect,0xfaa0,0xf00ff,m_state&ODS_SELECTED);
                DrawItemText(m_dc,m_str,m_rect);
                DrawMenuIcon(m_dc,m_rect,m_itemicon);
                break;
            }
        }
    }
}
/*************************************************************
功能描述: 绘制菜单项文本

参数说明: m_pdc标识画布对象,str标识菜单文本,m_rect标识菜单区域
*************************************************************/
void CIconMenu::DrawItemText(CDC* m_pdc,LPSTR str,CRect m_rect)
{
    m_rect.DeflateRect(20,0);
    m_pdc->DrawText(str,m_rect,DT_SINGLELINE|DT_VCENTER|DT_LEFT);
}

void CIconMenu::DrawTopMenu(CDC* m_pdc,CRect m_rect,BOOL m_selected )
{
    if (m_selected)
    {
        m_pdc->SelectStockObject(BLACK_PEN);
        m_pdc->Rectangle(&m_rect);
        m_rect.DeflateRect(1,1);
        m_pdc->FillSolidRect(m_rect,RGB(150, 185, 255));
    }
    else
    {
        CRect rect;
        AfxGetMainWnd()->GetClientRect(rect);
        rect.top = m_rect.top;
        rect.bottom = m_rect.bottom;
        rect.left= 360;
        rect.right +=4;
        //CRect c_rect(m_rect);
        m_pdc->FillSolidRect(&rect,RGB(200,187, 255));
        m_pdc->FillSolidRect(&m_rect,RGB(200,187, 255));
        //m_pdc->SelectStockObject(BLACK_PEN);
        //m_pdc->FillSolidRect(m_rect,RGB(100, 185, 255));
    }
}

void CIconMenu::DrawSeparater(CDC* m_pdc,CRect m_rect)
{
    if (m_pdc != NULL)
    {
        m_pdc->Draw3dRect(m_rect,RGB(255,0,0),RGB(0,0,255));
    }
}

void CIconMenu::DrawComMenu(CDC* m_pdc,CRect m_rect,COLORREF m_fromcolor,COLORREF m_tocolor, BOOL m_selected )
{
    if (m_selected)
    {
        
        m_pdc->Rectangle(m_rect);     
        m_rect.DeflateRect(1,1);
        int r1,g1,b1; 
        //读取渐变起点的颜色值
        r1 = GetRValue(m_fromcolor);
        g1 = GetGValue(m_fromcolor);
        b1 = GetBValue(m_fromcolor);
        int r2,g2,b2;


        //读取渐变终点的颜色值
        r2 = GetRValue(m_tocolor);
        g2 = GetGValue(m_tocolor);
        b2 = GetBValue(m_tocolor);

        float  r3,g3,b3;//菜单区域水平方向每个点RGB值应该变化的度(范围)
        
        r3 = ((float)(r2-r1)) / (float)(m_rect.Height());
        g3 = (float)(g2-g1)/(float)(m_rect.Height());            
        b3 = (float)(b2-b1)/(float)(m_rect.Height());
        
        COLORREF r,g,b;//菜单区域水平方向每个点的颜色值
        CPen* m_oldpen ;

        for (int i = m_rect.top;i<m_rect.bottom;i++)
        {
            r = r1+(int)r3*(i-m_rect.top);
            g = g1+(int)g3*(i-m_rect.top);
            b = b1+ (int)b3*(i-m_rect.top);
            
            CPen m_pen (PS_SOLID,1,RGB(r,g,b));
            m_oldpen = m_pdc->SelectObject(&m_pen);
            m_pdc->MoveTo(m_rect.left,i);
            m_pdc->LineTo(m_rect.right,i);
        }

        m_pdc->SelectObject(m_oldpen);
    }
    else
    {
        m_pdc->FillSolidRect(m_rect,RGB(0x000000F9, 0x000000F8, 0x000000F7));
    }
}

void CIconMenu::DrawMenuIcon(CDC* m_pdc,CRect m_rect,int m_icon )
{
    m_imagelist.Draw(m_pdc,m_icon,CPoint(m_rect.left+2,m_rect.top+4),ILD_TRANSPARENT);
}



2008031901.jpg



本文转自Phinecos(洞庭散人)博客园博客,原文链接:http://www.cnblogs.com/phinecos/archive/2008/03/19/1112958.html,如需转载请自行联系原作者
目录
相关文章
|
5月前
|
开发框架 数据可视化 C#
UGUI系列-鼠标移动到按钮上显示信息(Unity3D)
有时候图标不能很好的说明这个功能的解释,就需要一些说明性文字显示。就比如可以在鼠标移动到UI上面的时候显示文字。 那么如何在UGUI上,鼠标移动上去显示文字说明呢。 大家都知道,当鼠标移动到button按钮上面的时候会出现变化,主要是button这个组件在控制
QT 标题栏隐藏可拖拽
这个也是我网上找到了 为了方便,记录一下
134 0
可折叠侧边栏菜单
在线演示 本地下载
1370 0
半透明菜单导航
在线演示 本地下载
815 0
UWP 取消GridView、ListView鼠标选中、悬停效果
原文:UWP 取消GridView、ListView鼠标选中、悬停效果 因为经常碰到ListView或者ListBox之类的选中、鼠标悬停样式和自己设置的主题颜色不搭,这时就需要改变这些样式了. 而这里我通过ListView来说明,大致思路其实就是重新定义Item的Template。
1198 0
|
索引 Windows 程序员