NM_CUSTOMDRAW的Listview项目正在闪烁

时间:2014-08-04 07:01:38

标签: listview winapi colors gdi custom-draw

大家。 http://i.stack.imgur.com/ugfY4.jpg

我有这种列表视图颜色编辑器,每个项目代表单独的颜色。 因此,用户单击COLOR_CODE子项,颜色选择器更新为所选的HSV值,然后用户拖动托盘上的颜色选择器光标,COLOR_CODE子项应实时更新以及颜色ID文本。 大多数情况下,更新表现良好且流畅,但有时它只是以一种方式闪烁 - ---- 它以一种非常闪烁的方式发生,好像没有时间快速绘制它。

我开始搜索并找到了很多帖子,都导致了双重缓冲。 好的,我在列表视图中启用了DOUBLE BUFFERING,就像那样

case WM_INITDIALOG:
ListView_SetExtendedListViewStyle(GetDlgItem(hDlg,id_listview),LVS_EX_DOUBLEBUFFER);

也试过这种方式

SendDlgItemMessage(hWnd,id_listview,LVM_SETEXTENDEDLISTVIEWSTYLE,NULL,(LPARAM)LVS_EX_DOUBLEBUFFER);

但它没有帮助。 这是我的自定义绘图程序 它基本上采用ID子项中写入的任何字符串 - 例如0xffb400 并将其转换为COLORREF,然后将subItem 2的BG颜色设置为生成的颜色;

case WM_NOTIFY:
if(((LPNMHDR)lParam)->code == NM_CUSTOMDRAW)
        {
            SetWindowLong(hDlg, DWL_MSGRESULT, (LONG)ProcessCustomDraw(lParam,hDlg));
            return TRUE;
        }
        break;

LRESULT colorEditor::ProcessCustomDraw (LPARAM lParam,HWND hDlg)
{
    LPNMLVCUSTOMDRAW lplvcd = (LPNMLVCUSTOMDRAW)lParam;

    switch(lplvcd->nmcd.dwDrawStage) 
    {
        case CDDS_PREPAINT:
            return CDRF_NOTIFYITEMDRAW;
        case CDDS_ITEMPREPAINT: 
            return CDRF_NOTIFYSUBITEMDRAW;
        case CDDS_ITEMPREPAINT|CDDS_SUBITEM: //Before an item is drawn//
            if (lplvcd->iSubItem==2)
            {
                item_redraw.iItem=lplvcd->nmcd.dwItemSpec;
                SendDlgItemMessageA(hWnd,id_listview,LVM_GETITEM,0,(LPARAM)&item_redraw);
                lplvcd->clrTextBk = colorrefFromString(item_redraw.pszText);
            }
            return CDRF_NEWFONT;
    }
    return CDRF_DODEFAULT;
}

万一你认为colorrefFromString是我提供它的列表:

COLORREF colorEditor::colorrefFromString(wchar_t *color)
{
    COLORREF res_color;
    unsigned short i=0,di=0;
    int digits[6];
    int h_digits[3];
    if (color[i]=='0'&&(color[i+1]=='x'||color[i+1]=='X')) i=2;
    int ix=0;

    while (color[(ix++)+i]!='\0'){}
    if (--ix!=5) while((ix++)<5)
        digits[di++]=0;

    while (color[i]!='\0')
    {
        if (color[i]>47&&color[i]<58) digits[di++]=color[i]-48;
        else if (color[i]>64&&color[i]<71) digits[di++]=color[i]-65+10;
        else if (color[i]>96&&color[i]<103) digits[di++]=color[i]-97+10;
        i++;
    }
    h_digits[0]=digits[0]*16+digits[1];
    h_digits[1]=digits[2]*16+digits[3];
    h_digits[2]=digits[4]*16+digits[5];
    res_color=0x00000000|(h_digits[2]<<16)|(h_digits[1]<<8)|h_digits[0];
    return res_color;
}

现在问题: 为什么我会闪烁?

1 个答案:

答案 0 :(得分:0)

第一眼看,我看不出闪烁的真正原因。

但是试着自己动手绘制所有这些。可能会有效。

只需计算颜色。使用给定的dc调用项目矩形上的FillRect并返回CDRF_SKIPDEFAULT。