当ListView的ViewStyle属性为vsReport时,当点击列标头排序时如何在列标头文字右边出现箭头图标?

解决方案 »

  1.   

    示范代码:
    procedure Tfrmmain.ListView1ColumnClick(Sender: TObject;
      Column: TListColumn);
    var
      intTmp:integer;
    begin
      //清空所有表头的图片
      for intTmp := 0 to ListView1.Columns.Count - 1 do
        ListView1.Columns[intTmp].ImageIndex := -1;
      
      //指定当前被点击的表头的图片
      Column.ImageIndex :=1;
    end;
      

  2.   

    to  ShanShiMin(Delphi+C#=我的最爱) 
    你的办法是可行,怎么把图片出现在字的右边
      

  3.   

    一定要右边?那你只能利用canvas自绘了。
      

  4.   

    typeTListView1 = class(TListView)private  FaToz :Boolean;  FoldCol :Integer;  FPicture :TPicture;  FHeaderFont:TFont;  procedure SetHeaderFont(Value:TFont);  procedure SetHeaderStyle(phd:PHDNotify);  procedure DrawHeaderItem(pDS:PDrawItemStruct);  procedure SetPicture(Value: TPicture);  procedure PictureChanged(Sender: TObject);  procedure LVCustomDraw(Sender:TCustomListView;const ARect:TRect;var DefaultDraw:Boolean);  procedure DrawBack;protected  procedure WndProc(var Message : TMessage); override;public  constructor Create(AOwner: TComponent); override;  destructor Destroy; override;  procedure SortColumn(Column: TListColumn);published  property BackPicture: TPicture read FPicture write SetPicture;  property HeaderFont: TFont read FHeaderFont write SetHeaderFont;end;  说明:a). 在published段我们定义了两个属性。背景图属性BackPicture,其数据类型是TPicture;表头字体属性HeaderFont,其数据类型是Tfont;b). 为了读/写BackPicture属性的值,在private段分别定义了它的私有数据FPicture和属性的写方法SetPicture;同理,在private段为HeaderFontn属性分别定义了它的私有数据FHeaderFont和属性的写方法SetHeaderFont;c). 在public段重载了TListView的构造函数和析构函数;d). 在 protected段重载了TListView的WndProc过程;e). 为了能在设计期间动态改变视图的背景图,我们自定义了二个事件响应过程,PictureChanged和LVCustomDraw。PictureChanged是背景图属性BackPicture的私有数据FPicture(TPicture)的OnChange事件响应过程,设计期间当我们通过Delphi的Object Inspector面板改变BackPicture的值时,将产生OnChang事件而执行该过程重绘列表视图(过程就是这样写的),这又将产生视图的OnCustomDraw事件而执行我们自定义的LVCustomDraw事件响应过程,也即LVCustomDraw是列表视图的OnCustomDraw事件响应过程;f). 在protected段重载的WndProc过程用于捕获Windows消息,它是我们完成这个自定义列表视图的核心所在,所需捕获的消息和作用在下面的代码中以注释的形式给出。g). 我们必须手工在单元文件的uses子句后加上CommCtrl。  二、编写控件的过程体    Delphi自动生成的 procedure Register可以不理它。我们在它的过程体之后,在end.(注意符号“.”)之前手工加上以下代码,完成我们在上面定义的全部过程的过程体编写(这里我们没有定义有函数原型)://============== 构造函数 ===================================constructor TListView1.Create(AOwner: TComponent);begin  inherited Create(AOwner);//继承  FHeaderFont:=TFont.Create;  FPicture:=TPicture.Create;  FPicture.OnChange:=PictureChanged;  OnCustomDraw:=LVCustomDraw;end;//============== 析构函数 ===================================destructor TListView1.Destroy;begin  FPicture.Free;  FHeaderFont.Free;  inherited Destroy;//继承end;//============== 设置表头字体 ===============================procedure TListView1.SetHeaderFont(Value:TFont);begin  //转换表头字体设置,将值给FHeaderFomt私有数据域,并重绘表头区域  if FHeaderFont <> Value then begin    FHeaderFont.Assign(Value);    InvalidateRect(GetDlgItem(Handle, 0),nil,true);//调用Windows API(二个函数均是)  end;end;//============== 设置背景图 =================================procedure TListView1.SetPicture(Value: TPicture);begin  //转换背景图设置,将值赋给FPicture私有数据域  if FPicture <> Value then    FPicture.Assign(Value);end;//============== TPicture的OnChange事件响应过程 ==============procedure TListView1.PictureChanged(Sender: TObject);begin  //重绘列表视图  Invalidate;end;//============== TListView的OnCustomDraw事件响应过程==========procedure TListView1.LVCustomDraw(Sender: TCustomListView; const ARect: TRect; var DefaultDraw: Boolean);begin  if (FPicture.Graphic<>nil)then begin    DrawBack;//绘制背景图    SetBkMode(Canvas.Handle,TRANSPARENT);//调用Windows API,将画布的背景设为透明模式    ListView_SetTextBKColor(Handle,CLR_NONE);//调用Windows API,将Item的文本背景设为透明  end;end;//============== 绘制背景图 ==================================procedure TListView1.DrawBack;var x,y,dx: Integer;begin  x:=0;  y:=0;  if Items.Count>0 then begin    if ViewStyle = vsReport then x:=TopItem.DisplayRect(drBounds).Left    else x:=Items[0].DisplayRect(drBounds).Left;    y:=Items[0].DisplayRect(drBounds).Top-2;  end;  dx:=x;  while y<=ClientHeight do begin    while x<=ClientWidth do begin      Canvas.Draw(x,y,FPicture.Graphic);      inc(x,FPicture.Graphic.Width);    end;    inc(y,FPicture.Graphic.Height);    x:=dx;  end;end;//====== Windows 消息应答 ====================================procedure TListView1.WndProc(var Message : TMessage);var    pDS :PDrawItemStruct;    phd :PHDNotify;begin    inherited WndProc(Message);//继承    with Message do        case Msg of            WM_DRAWITEM :            begin //重绘列表项时               pDS := PDrawItemStruct(Message.lParam);               //在PDrawItemStruct数据结构中有我们需要的数据               if pDS.CtlType<>ODT_MENU then begin                   DrawHeaderItem(pDS);                   Result := 1;              end;           end;
      

  5.   


               WM_NOTIFY:           begin              phd := PHDNotify(Message.lParam);              //在PHDNotify数据结构中有我们需要的数据              if (phd.Hdr.hwndFrom = GetDlgItem(Handle, 0)) then              Case phd.Hdr.code of                //当单击表头时                HDN_ITEMCLICK,HDN_ITEMCLICKW:                begin                    SortColumn(Columns.Items[phd.item]);                    InvalidateRect(GetDlgItem(Handle, 0), nil, true);//调用Windows API                end;                //当拖动或改变表头时                HDN_ENDTRACK,HDN_ENDTRACKW,HDN_ITEMCHANGED:                begin                    SetHeaderStyle(phd);                    InvalidateRect(GetDlgItem(Handle, 0), nil, true);//调用Windows API                end;              end;          end;      end;end;//=====================================================================var AtoZOrder: Boolean;function CustomSortProc(Item1, Item2: TListItem; ParamSort: Integer): Integer; stdcall;begin//自定义TListView的排序函数类型TLVComparecase ParamSort of  0://主列排序      if AtoZOrder then         Result:=lstrcmp(PChar(TListItem(Item1).Caption), PChar(TListItem(Item2).Caption))      else         Result:=-lstrcmp(PChar(TListItem(Item1).Caption), PChar(TListItem(Item2).Caption));  else //子列排序      if(AtoZOrder) then         Result:=lstrcmp(PChar(TListItem(Item1).SubItems[ParamSort]),                       PChar(TListItem(Item2).SubItems[ParamSort-1]))      else         Result:=-lstrcmp(PChar(TListItem(Item1).SubItems[ParamSort-1]),                       PChar(TListItem(Item2).SubItems[ParamSort-1]));  end;end;//====== 可在外部调用的排序方法 ===================================procedure TListView1.SortColumn(Column: TListColumn);begin    //调用TListView的CustomSort函数,按列排序    if FOldCol = Column.Index then        FaToz:=not FAtoZ     else       FOldCol:=Column.Index;    AtoZOrder:= FaToz;    CustomSort(@CustomSortProc, Column.Index);end;//====== 绘制表头文本和图形 =======================================procedure TListView1.DrawHeaderItem(pDS :PDrawItemStruct);var   tmpCanvas :TCanvas;   tmpLeft :Integer;begin   tmpCanvas := TCanvas.Create;   tmpCanvas.Font := FHeaderFont;   tmpCanvas.Brush.Color := clBtnFace;   //重绘文字   tmpCanvas.Handle:=pDS.hDC;   tmpCanvas.Brush.Style:=bsClear;   tmpCanvas.TextOut(pDS^.rcItem.Left+6,pDS^.rcItem.Top+2,Columns[pDS^.itemID].Caption);   //绘制箭头   if (abs(pDS^.itemID) <> FOldCol) then Exit;     with tmpCanvas do        with pDS^.rcItem do        begin          tmpLeft:=TextWidth(Columns[pDS^.itemID].Caption)+Left+15;          if FAtoZ then begin //画箭头向上          Pen.Color := clBtnHighlight;          MoveTo(tmpLeft, Bottom - 5);          LineTo(tmpLeft + 8, Bottom - 5);          Pen.Color := clBtnHighlight;          LineTo(tmpLeft + 4, Top + 5);          Pen.Color := clBtnShadow;          LineTo(tmpLeft, Bottom - 5);        end else begin //画箭头向下          Pen.Color := clBtnShadow;          MoveTo(tmpLeft, Top + 5);          LineTo(tmpLeft + 8, Top + 5);          Pen.Color := clBtnHighlight;          LineTo(tmpLeft + 4, Bottom - 5);          Pen.Color := clBtnShadow;          LineTo(tmpLeft, Top + 5);        end;      end;   tmpCanvas.Free;end;//======== 设置表头样式 ===============================================procedure TListView1.SetHeaderStyle(phd:PHDNotify);var  i :integer;  hdi :THDItem;begin     for i := 0 to Columns.Count - 1 do   begin     hdi.Mask:= HDF_STRING or HDI_FORMAT;     hdi.fmt := HDF_STRING or HDF_OWNERDRAW;//设置表头样式为自绘式     Header_SetItem(phd.Hdr.hwndFrom ,i,hdi);//调用Windows API   end;//注意:如果不调用此过程,那么我们在前面绘制的图形将不能被清除掉end;//=====================================================================end.三、安装自定义组件    再次提醒:一定要在uses子句后手工加上CommCtrl!    检查确认无误后选择Delphi菜单的Component/Install Component选项,在Unite file name编辑框中确认你的文件路径和名称后按OK按钮,Delphi将编译安装该组件。    如果你完全按本文步聚进行,对Delphi生成的默认值不进行修改的话,在编译安装无误后,你可以在Delphi组件标签页的Samples标签页中找到一个图标和TListView一样的列表视图。新建一个工程并将这个我们自义的列表视图放置在Form上,其默认的名称是ListView11,此时你看到这个列表视图的外观和Delphi提供的TListView放置在Form上时的外观一样,但是我们却可以在Delphi的Object Inspector面板上找到BackPicture属性和HeaderFont属性,二者的设置方法和Delphi通常的图形属性和字体属性的设置方法一样。当我们将它的ViewStyle属性设为vsReport、并设了列和列的Caption文本时,可以通过HeaderFont这个我们新增的属性单独改变表头的字体。当然你也可以进一步修改,给表头再增加一个背景色属性等等。四、对PDrawItemStruct数据结构和PHDNotify数据结构的说明    (仅为说明数据定义而列出,和Delphi的原定义略有出入)    PDrawItemStruct在Delphi的Windows.pas文件中定义如下:            PDrawItemStruct = ^TDrawItemStruct;            tagDRAWITEMSTRUCT = packed record               CtlType: UINT;               CtlID: UINT;               itemID: UINT;               itemAction: UINT;               itemState: UINT;               hwndItem: HWND;               hDC: HDC;               rcItem: TRect;               itemData: DWORD;            end;            TDrawItemStruct = tagDRAWITEMSTRUCT;            DRAWITEMSTRUCT = tagDRAWITEMSTRUCT;        而关于DRAWITEMSTRUCT的解释可参见Delphi帮助文件(或微软)的Win32 Programmer's Reference。        PHDNotify在Delphi的CommCtrl.pas文件中定义如下:        tagNMHEADERA = packed record             Hdr: TNMHdr;             Item: Integer;             Button: Integer;             PItem: PHDItemA;        end;        PHDNotifyA = ^THDNotifyA;        PHDNotify = PHDNotifyA;        THDNotifyA = tagNMHEADERA;