CListCtrl行高的修改

CListCtrl行高的修改ide

用一個空白Image撐起來就能夠了
CImageList   m_l;
m_l.Create(1,24,TRUE|ILC_COLOR32,1,0);
m_list.SetImageList(&m_l,LVSIL_SMALL);函數

我也認爲重載CListCtrl::DrawItem函數能自繪item,但實際上子類的DrawItem死活沒反應。不知道怎麼回事。style已設置好了的。奇怪!字體

   誰能告訴我爲何?this

 

=============================code

 

解決方案:orm

   1. 設置List Control的屬性 Owen Draw Fixed.//我RALF曾經試過,直接用鼠標拉個控件上去勾選上不行,除非本身建立m_lst.Create(LVS_OWNERDRAWFIXED | WS_VISIBLE  |LVS_REPORT  , CRect(40,80,670,491),this,UINT(1701));不然許多虛函數都不會運行,包括MeasureItem消息也不產生。繼承

   2. 自定義CMyListCtrl, 繼承於CListCtrl,並重載CListCtrl::DrawItem.
         必須重載DrawItem函數,而不能本身處理WM_DRAWITEM,不然MFC處理時運行到CListCtrl::DrawItem會抱錯。( 此函數的內容只有一條語句: ASSERT(FALSE),因此,堅定不能運行^_^)(轉註:WM_DRAWITEM消息貌似也是發送給Control的Owner的。List Control接收不到該消息。)圖片

   3. 爲List Control所在的對話框添加對WM_MEASUREITEM消息的處理OnMeasureItem。在響應過程當中修改結構中的itemHeight參數。get

注意: 不能簡單地在CMyListCtrl中響應WM_MEASUREITEM消息,緣由很簡單,它根本收不到此消息。若是要更好的實現,能夠提供一個CMyListCtrl::MeasureItem的函數,在對話框的消息OnMeasureItem中,調用此方法。it

如下DrawItem實現片段:


void CMyListCtrl::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
{
TCHAR lpBuffer[256];

LV_ITEM lvi;

lvi.mask = LVIF_TEXT | LVIF_PARAM ;
lvi.iItem = lpDrawItemStruct->itemID ;
lvi.iSubItem = 0;
lvi.pszText = lpBuffer ;
lvi.cchTextMax = sizeof(lpBuffer);
VERIFY(GetItem(&lvi));

LV_COLUMN lvc, lvcprev ;
::ZeroMemory(&lvc, sizeof(lvc));
::ZeroMemory(&lvcprev, sizeof(lvcprev));
lvc.mask = LVCF_WIDTH | LVCF_FMT;
lvcprev.mask = LVCF_WIDTH | LVCF_FMT;

for ( int nCol=0; GetColumn(nCol, &lvc); nCol++)
{
   if ( nCol > 0 )
   {
    // Get Previous Column Width in order to move the next display item
    GetColumn(nCol-1, &lvcprev) ;
    lpDrawItemStruct->rcItem.left += lvcprev.cx ;
    lpDrawItemStruct->rcItem.right += lpDrawItemStruct->rcItem.left ;
   }

   // Get the text
   ::ZeroMemory(&lvi, sizeof(lvi));
   lvi.iItem = lpDrawItemStruct->itemID;
   lvi.mask = LVIF_TEXT | LVIF_PARAM;
   lvi.iSubItem = nCol;
   lvi.pszText = lpBuffer;
   lvi.cchTextMax = sizeof(lpBuffer);
   VERIFY(GetItem(&lvi));

   CDC* pDC;
   pDC = CDC::FromHandle(lpDrawItemStruct->hDC);

   if ( lpDrawItemStruct->itemState & ODS_SELECTED )
   {
    pDC->FillSolidRect(&lpDrawItemStruct->rcItem, GetSysColor(COLOR_HIGHLIGHT)) ;
    pDC->SetTextColor(GetSysColor(COLOR_HIGHLIGHTTEXT)) ;
   }
   else
   {
    pDC->FillSolidRect(&lpDrawItemStruct->rcItem, GetSysColor(COLOR_WINDOW)) ;
    pDC->SetTextColor(GetSysColor(COLOR_WINDOWTEXT)) ;
   }
 
   pDC->SelectObject(GetStockObject(DEFAULT_GUI_FONT));

   UINT   uFormat    = DT_LEFT ;

   ::DrawText(lpDrawItemStruct->hDC, lpBuffer, strlen(lpBuffer),
    &lpDrawItemStruct->rcItem, uFormat) ;

   pDC->SelectStockObject(SYSTEM_FONT) ;
}
}

以上代碼來自codeproject: http://www.codeproject.com/listctrl/changerowheight.asp

其上有一種解決方案以下,第1,2步相同,最後則以下處理:

   3. 在CMyListCtrl的MESSAGE_MAP中手動添加以下宏: ON_WM_MEASUREITEM_REFLECT()

    4. 重載CMyListCtrl::MeasureItem函數。一樣要注意,並非給CMyListCtrl添加消息處理函數。

 

備註: 此方法一樣適用於Combo Box, List Control, Menu

網上修改CListCtrl項高度的方法通常是擴大字體,及用圖片將項高度撐大.

這兩種方法雖然簡單,可是效果卻不是很理想.一種比較理想的方法是自畫CListCtrl,不過方法相對來講比較複雜.

要修改CListCtrl的列表項高度,咱們須要本身添加 MeasureItem 的消息響應函數,對應的消息是 WM_MEASUREITEM+WM_REFLECT_BASE, 而不是 WM_MEASUREITEM.在CListBox裏咱們能夠直接在 ClassWizard 裏將此消息響應添加進 class 裏,可是 CListCtrl 默認是沒有這個消息響應的,咱們須要手動添加它(注意,這裏不是 WM_MEASUREITEM. CListCtrl 僅有 WM_MEASUREITEM, 對應的函數爲 OnMeasureItem).

爲了響應這個消息,咱們還須要給列表加上 LVS_OWNERDRAWFIXED 風格.能夠在 Create 列表的時候添加,也能夠在 PreCreateWindow 虛函數中添加.

添加 MeasureItem 消息響應函數,首先咱們須要在類的頭文件中添加:

afx_msg void MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct);

來聲明此消息響應函數;

而後在cpp的消息響應宏中添加:

ON_WM_MEASUREITEM_REFLECT()

最後本身創建 MeasureItem 的函數定義:

/////////////////////////////////////////////////////////////////////////////
// CListEx message handlers

void CListEx::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct)
{
if( m_nItemHeight > 0 )
   lpMeasureItemStruct->itemHeight = m_nItemHeight;
}

其中 m_nItemHeight 是我在頭文件中聲明的一個成員變量,用於從外部修改列表項高度.

而後咱們添加一個方法,便於從外部直接修改列表項高度:

//設置行高
void SetItemHeight(UINT nHeight);

而後是該方法的定義:

//設置行高
void CListEx::SetItemHeight(UINT nHeight)
{
m_nItemHeight = nHeight;

CRect rcWin;
GetWindowRect(&rcWin);
WINDOWPOS wp;
wp.hwnd = m_hWnd;
wp.cx = rcWin.Width();
wp.cy = rcWin.Height();
wp.flags = SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOZORDER;

SendMessage(WM_WINDOWPOSCHANGED, 0, (LPARAM)&wp);
}

這個方法的最後,使用了 SendMessage 發送 WM_WINDOWPOSCHANGED 消息讓 CListCtrl 進入 MeasureItem 的消息響應函數,對列表高度進行修改.

由於咱們這裏使用了列表的自繪風格,所以列表項須要本身繪製.

首先在類的聲明中添加 DrawItem 虛函數聲明:

virtual void DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct);

而後自畫 CListCtrl:

void CListEx::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
{
int    nItem = lpDrawItemStruct->itemID;
CDC*   pDC   = CDC::FromHandle(lpDrawItemStruct->hDC);

......

CRect   rcBound, rcLabel, rcIcon;
//得到列表項圖標,標籤,及項的區域
GetItemRect ( nItem, rcIcon, LVIR_ICON );
GetItemRect ( nItem, rcLabel, LVIR_LABEL );
GetItemRect ( nItem, rcBound, LVIR_BOUNDS );

......

}

如今這個 CListCtrl 的重載類就支持自定義列表項高度了.  

相關文章
相關標籤/搜索