ComboBox控件自绘操作

一:

1、设置ComboBox控件的属性:Has Strings = True, Owner Draw = variable或fixed,Tyle = Drop List。

2、建立一个类:CMyComboBox,派生自:CComboBox,绑定关于CMyComboBox类的控件变量,并声明一个CImageList对象:

// 构造
public:
	CImageList m_imgList;  //声明CImageList对象
	CMyComboBox m_ComboBox;//绑定控件变量

	//......

说明: fixed属性表明下拉列表每项的高宽由系统自己设定;variable属性表示下拉列表每项的高宽由用户在MeasureItem回调函数中设定。

二、派生类中相关函数的定义和声明:

public:
	//保存ComboBox中每一项的文字和对应的图标索引
	struct ComboData{
		UINT nImageIndex;
		CString strText;
	};

	CImageList* m_ImgList;
	int m_nItemHeight;                    //用来设置下拉列表项每一项的高度
	CImageList* SetImageList(CImageList* pImageList);//加载创建的图标列表
	int AddString(LPCTSTR strText,int nImageInxex);//加载每一项的文字和对应的图标索引
	int GetLBText(int nItemIndex,CString& strText);//根据项索引获取对应的文字
	int GetLBImage(int nItemIndex,int& nImageIndex);//根据项索引获取对应的图标索引
	void ResetContent();//销毁申请的内存空间
	virtual void MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct);//重写虚函数
	virtual void DrawItem(LPDRAWITEMSTRUCT lpDIS);                    //重写虚函数    
	afx_msg void OnDestroy();

定义:

CMyComboBox::CMyComboBox()
{
	m_ImgList = NULL;
	m_nItemHeight = 20;
}

CMyComboBox::~CMyComboBox()
{
}


BEGIN_MESSAGE_MAP(CMyComboBox, CComboBox)
	ON_WM_DESTROY()
END_MESSAGE_MAP()



// CMyComboBox 消息处理程序
CImageList* CMyComboBox::SetImageList(CImageList* pImageList)
{
	CImageList* pOldImgList = m_ImgList;

	m_ImgList = pImageList;

	return pOldImgList;
}

int CMyComboBox::AddString(LPCTSTR strText,int nImageInxex)
{
	//int nItem = CComboBox::AddString(strText);
	int nItem = CComboBox::InsertString(nImageInxex,strText);//插入一条文本

	ComboData* pData = new ComboData;

	if(pData == NULL) return -1;

	pData->strText = strText;

	pData->nImageIndex = nImageInxex;

	SetItemData(nItem,(DWORD)pData);

	return nItem;
}

int CMyComboBox::GetLBText(int nItemIndex,CString& strText) //获取当前绘制项的文本
{
	ComboData* pData = (ComboData*)GetItemData(nItemIndex);

	if(pData == NULL) return 0;

	strText = pData->strText;

	return 1;
}

int CMyComboBox::GetLBImage(int nItemIndex,int& nImageIndex) //获取当前绘制项的图像
{
	ComboData* pData = (ComboData*)GetItemData(nItemIndex);

	if(pData == NULL) return 0;

	nImageIndex = pData->nImageIndex;

	return 1;
}

void CMyComboBox::ResetContent()//释放所申请的内存
{
	int nItemIndex = GetCount();

	while (nItemIndex--)
	{
		ComboData* pData = (ComboData*)GetItemData(nItemIndex);

		if (pData)
		{
			delete pData;
			pData = NULL;
		}
	}
	CComboBox::ResetContent();
}

void CMyComboBox::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct)
{
	// TODO:  添加您的代码以确定指定项的大小

	lpMeasureItemStruct->itemHeight = m_nItemHeight;
}

void CMyComboBox::DrawItem(LPDRAWITEMSTRUCT lpDIS)
{
	CDC dc;
	dc.Attach(lpDIS->hDC);//成对使用

	CRect ItemRect = lpDIS->rcItem;//获取当前项的矩形框大小

	int nItem = lpDIS->itemID;//获取当前项的索引

	if(nItem == -1) return;

	dc.SetBkMode(TRANSPARENT);

	if (lpDIS->itemState & ODS_SELECTED)//选中项设置
	{
		dc.SetTextColor(GetSysColor(COLOR_HIGHLIGHTTEXT));
		dc.FillSolidRect(&ItemRect,GetSysColor(COLOR_HIGHLIGHT));

		if(lpDIS->itemAction & ODA_FOCUS) dc.DrawFocusRect(&ItemRect);
	} 
	else//非选中项设置
	{
		dc.SetTextColor(GetSysColor(COLOR_WINDOWTEXT));
		dc.FillSolidRect(&ItemRect,RGB(200,0,0)/*GetSysColor(COLOR_WINDOW)*/);
	}

	
	//画文字
	CRect TextRect = ItemRect;

	CString strText;

	GetLBText(nItem,strText);//获取正在自绘项的文字

	TRACE("nItem = %d\r\n",nItem);

	TextRect.left = TextRect.left + 16 + 4;

	dc.DrawText(strText,&TextRect,DT_LEFT|DT_VCENTER|DT_SINGLELINE);
	
	//画图标
	CRect IconRect = ItemRect;

	int nImageIndex;

	GetLBImage(nItem,nImageIndex);

	//IconRect.right = m_nItemHeight;

	m_ImgList->Draw(&dc,nImageIndex,IconRect.TopLeft(),ILD_TRANSPARENT);

	dc.Detach();//成对使用
}

void CMyComboBox::OnDestroy()
{
	CComboBox::OnDestroy();

	ResetContent();
}

三、主对话框初始化函数中进行函数的调用操作:

BOOL CComboBox自绘Dlg::OnInitDialog()
{
	CDialogEx::OnInitDialog();

	SetIcon(m_hIcon, TRUE);			// 设置大图标
	SetIcon(m_hIcon, FALSE);		// 设置小图标

	//创建CImageList对象
	m_imgList.Create(16,16,ILC_COLOR32|ILC_MASK,6,6);

	//将导入的图标依次放入到CImageList对象中
	m_imgList.Add(AfxGetApp()->LoadIcon(IDI_ICON1));
	m_imgList.Add(AfxGetApp()->LoadIcon(IDI_ICON2));
	m_imgList.Add(AfxGetApp()->LoadIcon(IDI_ICON3));
	m_imgList.Add(AfxGetApp()->LoadIcon(IDI_ICON4));
	m_imgList.Add(AfxGetApp()->LoadIcon(IDI_ICON5));
	m_imgList.Add(AfxGetApp()->LoadIcon(IDI_ICON6));

	m_ComboBox.SetImageList(&m_imgList);//将CImageList指针传递给派生类

	m_ComboBox.AddString("北京",0);//添加文本和图标索引
	m_ComboBox.AddString("上海",1);
	m_ComboBox.AddString("广州",2);
	m_ComboBox.AddString("深圳",3);
	m_ComboBox.AddString("天津",4);
	m_ComboBox.AddString("重庆",5);

	m_ComboBox.SetCurSel(0);//设置选中项

	return TRUE;  
}



附上tagDRAWITEMSTRUCT结构的说明:

DRAWITEMSTRUCT 结构的定义如下:

  typedef struct tagDRAWITEMSTRUCT {

  UINT CtlType;

  UINT CtlID;

  UINT itemID;

  UINT itemAction;

  UINT itemState;

  HWND hwndItem;

  HDC hDC;

  RECT rcItem;

  ULONG_PTR itemData;

  } DRAWITEMSTRUCT, NEAR *PDRAWITEMSTRUCT, FAR *LPDRAWITEMSTRUCT ; 结构成员:

  成员:

  CtlType

   指定了控件的类型,其取值如下表所示。

  ODT_BUTTON :按钮控件

  ODT_COMBOBOX :组合框控件

  ODT_LISTBOX :列表框控件

  ODT_LISTVIEW :列表视图控件

  ODT_MENU :菜单项

  ODT_STATIC :静态文本控件

  ODT_TAB :Tab 控件

  CtlID

   指定了自绘控件的ID 值,而对于菜单项则不需要使用该成员

  itemID

  表示菜单项ID ,也可以表示列表框或者组合框中某项的索引值。对于一个空的列表框或组合框,该 成员的值为–1 。这时应用程序只绘制焦点矩形(该矩形的坐标由rcItem 成员给出)虽然此时控件中没有需要显示的项,但是绘制焦点矩形还是很有必要的,因为这样做能够提示用户该控件是否具有输入焦点。当然也可以设置 itemAction 成员为合适值,使得无需绘制焦点。

  itemAction

  指定绘制行为,其取值可以为下表中所示值的一个或者多个的联合。

  ODA_DRAWENTIRE :当整个控件都需要被绘制时,设置该值

  ODA_FOCUS :如果控件需要在获得或失去焦点时被绘制,则设置该值。此时应该检查itemState成员,以确定控件是否具有输入焦点。

  ODA_SELECT

  如果控件需要在选中状态改变时被绘制,则设置该值。此时应该检查itemState 成员,以确定控件是否处于选中状态。

  itemState

  指定了当前绘制操作完成后,所绘项的可见状态。例如,如果菜单项应该被灰色显示,则可以指定ODS_GRAYED 状态标志。其取值可以为下表中所示值的一个或者多个的联合。

  ODS_CHECKED :如果菜单项将被选中,则可设置该值。该值只对菜单项有用。

  ODS_COMBOBOXEDIT :在自绘组合框控件中只绘制选择区域。

  ODS_DEFAULT :默认值。

  ODS_DISABLED :如果控件将被禁止,则设置该值。

  ODS_FOCUS :如果控件需要输入焦点,则设置该值。

  ODS_GRAYED :如果控件需要被灰色显示,则设置该值。该值只在绘制菜单时使用。

  ODS_HOTLIGHT :Windows 98/Me, Windows 2000/XP: 如果鼠标指针位于控件之上,则设置该值,这时控件会显示高亮颜色。

  ODS_INACTIVE :Windows 98/Me, Windows 2000/XP: 表示没有激活的菜单项。

  ODS_NOACCEL :Windows 2000/XP: 控件是否有快速键盘。

  ODS_NOFOCUSRECT :Windows 2000/XP: 不绘制捕获焦点的效果。

  ODS_SELECTED :选中的菜单项。

  hwndItem

  指定了组合框、列表框和按钮等自绘控件的窗口句柄;如果自绘的对象时菜单项,则表示包含该菜单项的菜单句柄。

  hDC

  指定了绘制操作所使用的设备环境。

  rcItem

  指定了将被绘制的矩形区域。这个矩形区域就是上面hDC 的作用范围。系统会自动裁剪组合框、列 表框或按钮等控件的自绘制区域以外的部分。也就是说rcItem 中的坐标点(0 ,0 )指的就是控件的左上角。但是系统不裁剪菜单项,所以在绘制菜单项的时 候,必须先通过一定的换算得到该菜单项的位置,以保证绘制操作在我们希望的区域中进行。

  itemData

  对于菜单项,该成员的取值可以是由

  CMenu::AppendMenu 、

  CMenu::InsertMenu 或者

  CMenu::ModifyMenu

  等函数传递给菜单的值。

  对于列表框或这组合框,该成员的值可以为由

  ComboBox::AddString 、

  CComboBox::InsertString 、

  CListBox::AddString 或者

  CListBox::InsertString

  等传递给控件的值。

如果ctlType 的取值是ODT_BUTTON 或者ODT_STATIC, itemData 的取值为0 。


猜你喜欢

转载自blog.csdn.net/u012372584/article/details/78111980