MFC中ListCtrl,改变具体单元格的文本和背景颜色

该文章介绍了如何在MFC的ListCtrl控件中,通过处理NM_CUSTOMDRAW消息来改变特定单元格的文本和背景颜色。在OnNMCustomdrawList函数中,根据dwDrawStage判断绘制阶段,然后根据iSubItem识别列,对超出特定范围的数据(如年龄、身高)设置异常颜色,其余则保持默认颜色。
摘要由CSDN通过智能技术生成

MFC中ListCtrl,改变具体单元格的文本和背景颜色


下面展示一些 内联代码片


先要创建NM_CUSTOMDRAW的消息

// A code block
var foo = 'bar';
// An highlighted block
void CStudentDlg::OnNMCustomdrawList(NMHDR *pNMHDR, LRESULT *pResult)
{
	NMLVCUSTOMDRAW* pLVCD = reinterpret_cast<NMLVCUSTOMDRAW*>(pNMHDR);
	// TODO: Add your control notification handler code here
	*pResult = CDRF_DODEFAULT;


	if (CDDS_PREPAINT == pLVCD->nmcd.dwDrawStage)
	{
		*pResult = CDRF_NOTIFYITEMDRAW;
	}
	else if (CDDS_ITEMPREPAINT == pLVCD->nmcd.dwDrawStage)
	{
		// This is the notification message for an item. We'll request
		// notifications before each subitem's prepaint stage.

		*pResult = CDRF_NOTIFYSUBITEMDRAW;
	}//这上面的都是默认的不用理
	else if ((CDDS_ITEMPREPAINT | CDDS_SUBITEM) == pLVCD->nmcd.dwDrawStage)
	{
		COLORREF clrNewTextColor, clrNewBkColor;//当前单元格的文本颜色以及背景颜色
		int nItem = static_cast<int>(pLVCD->nmcd.dwItemSpec);//当前行
		CString strTemp = mList.GetItemText(nItem, pLVCD->iSubItem);//当前行的某一列的内容
		switch (pLVCD->iSubItem)//pLVCD->iSubItem表示当前列,等到所有列都遍历完才会进入下一行
		{
		case 3://表明处理到此行的第3列时
		{
				  int age = atoi((CT2A(strTemp.GetBuffer())));//字符串转int
				  if (age > 22 || age < 18)//当年龄超过范围,设置字体颜色异常
				  {
					  clrNewTextColor = RGB(255, 0, 0);
					  clrNewBkColor = RGB(255, 255, 255);
					  pLVCD->clrText = clrNewTextColor;
					  pLVCD->clrTextBk = clrNewBkColor;
				  }  
				  else //正常年龄范围为黑色文本
				  {
					  pLVCD->clrText = RGB(0, 0, 0);
					  pLVCD->clrTextBk = RGB(255, 255, 255);
				  }
				  *pResult = CDRF_DODEFAULT;
				  break;
		}
		case 4://表明处理到此行的第4列时
		{
				  int height = atoi((CT2A(strTemp.GetBuffer())));//字符串转int
				  if (height > 190 || height < 150)//当身高超过范围,设置字体颜色异常
				  {
					  clrNewTextColor = RGB(0, 0, 255);
					  clrNewBkColor = RGB(255, 255, 255);
					  pLVCD->clrText = clrNewTextColor;
					  pLVCD->clrTextBk = clrNewBkColor;
				  }
				  else //正常年龄范围为黑色文本
				  {
					  pLVCD->clrText = RGB(0, 0, 0);
					  pLVCD->clrTextBk = RGB(255, 255, 255);
				  }
				  *pResult = CDRF_DODEFAULT;
				  break;
		}
		default://默认为黑色文本
			pLVCD->clrText = RGB(0, 0, 0);
			pLVCD->clrTextBk = RGB(255, 255, 255);
			*pResult = CDRF_DODEFAULT;
			break;
		}
	}
}

该消息处理函数

下面的第3列和第四列就是对应上面case 3和case 4的处理
在这里插入图片描述

MFCLISTCTRL控件默认情况下是无法自动改变行高的,需要通过代码实现。以下是实现自动改变行高的方法: 1. 首先需要在列表控件的创建过程设置风格为LVS_REPORT,这可以在属性窗口设置或者在代码使用ModifyStyle函数设置。 2. 在列表控件的初始化过程,需要设置行高为所需的高度,可以使用SetItemHeight函数设置。 3. 在列表控件的绘制过程,需要计算每个单元格文本高度,并将行高设置为最高的文本高度。可以使用GetDC函数获取设备上下文,然后使用GetTextExtent函数计算文本高度。 4. 在列表控件的大小改变事件,需要重新计算每个单元格文本高度,并将行高设置为最高的文本高度。可以使用GetClientRect函数获取列表控件的客户端区域,然后根据客户端区域大小计算每个单元格的大小。 以下是示例代码: ```cpp // 设置列表控件风格为LVS_REPORT m_listCtrl.ModifyStyle(0, LVS_REPORT); // 设置行高为20 m_listCtrl.SetItemHeight(-1, 20); // 在绘制过程计算文本高度并设置行高 void CMyDlg::OnCustomDrawList(NMHDR* pNMHDR, LRESULT* pResult) { LPNMLVCUSTOMDRAW lpLVCustomDraw = reinterpret_cast<LPNMLVCUSTOMDRAW>(pNMHDR); // 判断绘制状态 switch (lpLVCustomDraw->nmcd.dwDrawStage) { case CDDS_PREPAINT: *pResult = CDRF_NOTIFYITEMDRAW; break; case CDDS_ITEMPREPAINT: { // 计算文本高度 CString strText = m_listCtrl.GetItemText(lpLVCustomDraw->nmcd.dwItemSpec, lpLVCustomDraw->iSubItem); CRect rectText(lpLVCustomDraw->nmcd.rc); CDC* pDC = CDC::FromHandle(lpLVCustomDraw->nmcd.hdc); pDC->DrawText(strText, rectText, DT_CALCRECT); // 设置行高为文本高度 int nHeight = rectText.Height() + 4; m_listCtrl.SetItemHeight(lpLVCustomDraw->nmcd.dwItemSpec, nHeight); *pResult = CDRF_DODEFAULT; } break; default: *pResult = CDRF_DODEFAULT; break; } } // 在大小改变事件重新计算文本高度并设置行高 void CMyDlg::OnSize(UINT nType, int cx, int cy) { CDialogEx::OnSize(nType, cx, cy); // 计算单元格大小 CRect rectClient; m_listCtrl.GetClientRect(rectClient); int nWidth = rectClient.Width() / m_listCtrl.GetHeaderCtrl()->GetItemCount(); int nHeight = 20; // 设置列宽和行高 for (int i = 0; i < m_listCtrl.GetHeaderCtrl()->GetItemCount(); i++) { m_listCtrl.SetColumnWidth(i, nWidth); } m_listCtrl.SetItemHeight(-1, nHeight); } ``` 以上代码,OnCustomDrawList函数处理列表控件的绘制过程,通过计算文本高度并设置行高实现自动改变行高。OnSize函数处理列表控件的大小改变事件,重新计算文本高度并设置行高。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值