近期把List Control扩展成Grid,参考了两篇关于SubItem Selection的文章,一篇来自
Piotr Szewczyk.. 采用NM_CUSTOMDRAW方法,另一篇来自
Muhammad Azam.
,采用LVS_OWNERDRAWFIXED。我个人偏好于NM_CUSTOMDRAW,因为NM_CUSTOMDRAW可以控制是否自绘,可以控制自绘区域(CDDS_ITEMPREPAINT和CDDS_SUBITEM),详细内容请参阅
Neat Stuff to do in List Controls Using Custom Draw
。
但两篇文章都存在一个问题:
DrawText
时总是用
DT_LEFT
,因为我为每个
Column
都设了
Alignment
,结果是每列都是
Align Left
。于是我用作了修改,
DrawText
时分别对应每个
Column
的
Alignment
,但是问题又出现了。
我用NM_CUSTOMDRAW方法,当移动Column Header时,如果该列的Alignment为Center或Right时,该列的内容就出现“花屏”现象,原因是作者为了防止闪烁,采用以下方法:
void CMyList::InvalidateCell(int nRow, int nCol)
{
//I add this function to reduce flickering
CRect rc;
if(nCol==0)
GetItemRect(nRow,&rc,LVIR_LABEL);
else
GetSubItemRect(nRow,nCol,LVIR_BOUNDS,rc);
InvalidateRect(&rc);
}
程序中保存两个变量m_nRow,m_nCol,当变量的值改变了就会调用InvalidateCell(int nRow, int nCol)。而当我拖动Column Header时不会调用InvalidateCell(int nRow, int nCol)。
解决方法当然是捕获拖动Column Header时的消息,查看了相关资料,相关的消息是HDN_ITEMCHANGED:拖动过程中不断收到
HDN_BEGINTRACK:拖到开始时
EDN_ENDTRACK:拖动结束时
那些消息是从CHeaderCtrl反馈回来的。(注:CHeaderCtrl有两种获得方法:一种是CListCtrl::GetHeaderCtrl(),一种是CListCtrl::GetDlgItem(0)。)
我的CMyList是继承自CListCtrl,不论我怎样设定风格,如CMyList->GetHeaderCtrl()->ModifyStyle(0, HDS_HOTTRACK),还是捕获不了HDN_ITEMCHANGED,其它消息也捕不了。
最后的办法只有设计一个类CMyHeaderCtrl继承自CHeaderCtrl,然后在CMyList的SubclassHeaderControl加入以下代码:
CHeaderCtrl* pHeader = GetHeaderCtrl();
if (pHeader)
{
VERIFY(m_HeaderCtrl.SubclassWindow(pHeader->m_hWnd));
}
因为ClassView中没有HDN_ITEMCHANGED所以要自己加
ON_NOTIFY_REFLECT(HDN_ITEMCHANGED, OnItemChanged)
以下是OnItemChanged的实现
void CMyHeaderCtrl::OnItemChanged(NMHDR * pNMHDR, LRESULT* pResult)
{
*pResult = 0;
CMyList* pzList = static_cast(GetParent());
ASSERT(pzList);
NMHEADER *pHdr = (NMHEADER*)pNMHDR;
int nCol;//The changed column
int topRow,bottomRow;//The visit top row and bottom row
nCol = pHdr->iItem;
if( pzList->Cols[nCol].getTextAlignFixed() == AlignLeft)
return;//因为AlignLeft不存在“花屏”现象
if( 0 == pzList->GetItemCount())
return;
topRow = pzList->GetTopIndex();
bottomRow = pzList->GetCountPerPage();
CRect rc1,rc2;
pzList->GetSubItemRect(topRow,nCol,LVIR_BOUNDS,rc1);
pzList->GetSubItemRect(bottomRow,nCol,LVIR_BOUNDS,rc2);
if( 0 == nCol)
{
rc1.left = 0;
rc2.left = 0;
rc1.right = pzList->GetColumnWidth(0);
rc2.right = rc1.right;
}
rc1.bottom = rc2.bottom;
pzList->InvalidateRect(&rc1);
TRACE("ItemChanged/n");
}
至此,虽然问题已解决,重画区域已减到最小,但也存在小小的闪烁,但比Invalidate好多了。
一些问题也是想不通:
1. 为什么继承的CListCtrl不能收到CHeaderCtrl的Trace消息?
2. 的Column为什么不会出现“花屏”现象?AlignLeft
今天收获很大,虽然最后也要动用到继承CHeaderCtrl,但也不失为好事,因为迟早也要继承CHeaderCtrl才能完成一些功能的,如禁止改变列宽,在列上画排序箭头。