SubItem Selection in List Control改进版

近期把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<CMyList*>(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才能完成一些功能的,如禁止改变列宽,在列上画排序箭头。
参考资料:
Matt Weagle Using the Header Control
 
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值