Display Custom Hints for TListView Sub Items

You use the TListView Delphi control to display a list of items in a fashion similar to how Windows Explorer displays files and folders. The items can be displayed in columns with column headers and sub-i items, or vertically or horizontally, with small or large icons.

ListView Item Hints?

The TListView exposes the Hint and ShowHint properties you use to set up whether the hint (tooltip) should be displayed for the control when the mouse hovers over it.

TListView  Sub Item Hints

TListView Sub Item Hints

In most situations, when using the list view, you will need custom hints to be displayed for every list view item. What's more, a custom hint for every list view sub item might be required.

ListView provides the OnInfoTip event which gets fired when the mouse is paused over an item in the list view.
By default, when hints are enabled (that is, when ShowHint is true), the list view displays the hint specified by its Hint property. OnInfoTip allows the list view to override this value to specify a hint that is specific to the item under the mouse.

When ViewStyle is set to vsReport the ListView displays each item on its own line with information (sub items) arranged in columns.

Unfortunately, the OnInfoTip is not fired when the mouse pauses over a sub-item.

To force a list view to display custom hints for items and sub items, you need to handle two events: OnInfoTip and OnMouseMove. Here's the implementation of the OnInfoTip event handler.

 procedure TLVHintsForm.ListView1InfoTip(Sender: TObject; Item: TListItem; var InfoTip: string) ;
 begin
   //Show the "full" item - with all the sub items
   InfoTip := InfoTip + #13#10 + item.SubItems[0] + #13#10 + item.SubItems[1];
 end;
 
To show custom hints for every sub-item, you need to handle the OnMouseMove event.
 uses CommCtrl ... 

procedure TLVHintsForm.ListView1MouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer) ; var pt: TPoint; li : TLIstItem; lvHitInfo: TLVHitTestInfo; hint : string; begin pt := ListView1.ScreenToClient(Mouse.CursorPos) ; li := ListView1.GetItemAt(pt.x, pt.y) ; //over a sub item? if li = nil then begin FillChar(lvHitInfo, SizeOf(lvHitInfo), 0) ; lvHitInfo.pt := pt; //over a sub item! if -1 <> ListView1.Perform(LVM_SUBITEMHITTEST, 0, LParam(@lvHitInfo)) then begin hint := Format('Name: %s, %s : %s',[ ListView1.Items[lvHitInfo.iItem].Caption, ListView1.Columns[lvHitInfo.iSubItem].Caption, ListView1.Items[lvHitInfo.iItem].SubItems[-1 + lvHitInfo.iSubItem]]) ; if hint <> Memo1.Lines[0] then begin Memo1.Lines.Insert(0, hint) ; //activate hint ListView1.Hint := hint; Application.ActivateHint(Mouse.CursorPos) ; end; end; end; end;

After getting the Mouse position (Mouse.CursorPos), using ScreenToClient we get the position of the mouse in ListView1's coordinates.

Sending the LVM_SUBITEMHITTEST message to the list view fills the (list view specific) TLVHitTestInfo record. This structure helps us find which list view item or subitem is at a given position.

If the mouse is over a sub-item, we activate the hint window by calling the Application.ActivateHint method.

That's it, really.

posted @ 2011-03-18 14:41  Max Woods  阅读(557)  评论(0编辑  收藏  举报