是否可以覆盖列表视图详细选择画图?看起来半透明的蓝色覆盖在项目上,就像在资源管理器窗口中一样。
我想在所选内容周围绘制一个轮廓,以指示选择。
有办法做到这一点吗?我们很欣赏这些例子。
发布于 2009-08-04 18:41:17
这是一个快速的工作示例,我之前一直在胡乱操作。
First helper结构和枚举。
[StructLayout(LayoutKind.Sequential)]
public struct DRAWITEMSTRUCT
{
public int CtlType;
public int CtlID;
public int itemID;
public int itemAction;
public int itemState;
public IntPtr hwndItem;
public IntPtr hDC;
public RECT rcItem;
public IntPtr itemData;
}
[StructLayout(LayoutKind.Sequential)]
public struct RECT
{
public int left;
public int top;
public int right;
public int bottom;
public int Width
{
get { return right - left; }
}
public int Height
{
get { return bottom - top; }
}
}
public enum ListViewDefaults
{
LVS_OWNERDRAWFIXED = 0x0400
}
public enum WMDefaults
{
WM_DRAWITEM = 0x002B,
WM_REFLECT = 0x2000
}现在创建自定义ListView并覆盖CreateParams和WndProc
public class CustomListView : ListView
{
protected override CreateParams CreateParams
{
get
{
CreateParams cp = base.CreateParams;
//add OwnerDraw style...i took this idea from Reflecting against ListView
// bit OR is very important, otherwise you'll get an exception
cp.Style |= (int)ListViewDefaults.LVS_OWNERDRAWFIXED;
return cp;
}
}
protected override void WndProc(ref Message m)
{
base.WndProc(ref m);
//if we are drawing an item then call our custom Draw.
if (m.Msg == (int)(WMDefaults.WM_REFLECT | WMDefaults.WM_DRAWITEM))
ProcessDrawItem(ref m);
}现在是最重要的part..the绘图。我是一个相当业余的绘画爱好者,但这应该会让你知道该怎么做。
private void ProcessDrawItem(ref Message m)
{
DRAWITEMSTRUCT dis = (DRAWITEMSTRUCT)Marshal.PtrToStructure(m.LParam, typeof(DRAWITEMSTRUCT));
Graphics g = Graphics.FromHdc(dis.hDC);
ListViewItem i = this.Items[dis.itemID];
Rectangle rcItem = new Rectangle(dis.rcItem.left, dis.rcItem.top, this.ClientSize.Width, dis.rcItem.Height);
//we have our rectangle.
//draw whatever you want
if (dis.itemState == 17)
{
//item is selected
g.FillRectangle(new SolidBrush(Color.Red), rcItem);
g.DrawString(i.Text, new Font("Arial", 8), new SolidBrush(Color.Black), new PointF(rcItem.X, rcItem.Y+1));
}
else
{
//regular item
g.FillRectangle(new SolidBrush(Color.White), rcItem);
g.DrawString(i.Text, new Font("Arial", 8), new SolidBrush(Color.Black), new PointF(rcItem.X, rcItem.Y+1));
}
//we have handled the message
m.Result = (IntPtr)1;
}这就是结果。

发布于 2009-08-05 06:03:00
与其他答案相比,.NET ListView更直接地支持所有者绘制。你甚至不需要子类化。将OwnerDraw设置为true,侦听DrawSubItem事件,然后在该事件中可以绘制您喜欢的任何内容。
一如既往,ObjectListView使这个过程变得更容易。有this page记录了如何做到这一点。如果你觉得你对你的用户很刻薄,你可以像这样装饰:

但是,如果您想要在单元格本身的边界之外绘制某些内容,则这些技术都不起作用。因此,如果您希望在与前一行和后一行重叠的整行周围绘制选择轮廓,则不能通过所有者绘制来实现此目的。每个单元格都是单独绘制的,并“拥有”它在屏幕上的一部分,抹去了所有已经存在的部分。
要做你所要求的事情,你必须截取自定义绘制的后期绘制阶段(而不是所有者绘制。Michael Dunn wrote a great introduction为CodeProject定制绘图)。您可以阅读所需的here。
我不想这么说,但最简单的答案是使用ObjectListView,创建一个装饰并安装它:
public void InitializeSelectionOverlay()
{
this.olv1.HighlightForegroundColor = Color.Black;
this.olv1.HighlightBackgroundColor = Color.White;
this.olv1.AddDecoration(new SelectedRowDecoration());
}
public class SelectedRowDecoration : IOverlay
{
public void Draw(ObjectListView olv, Graphics g, Rectangle r) {
if (olv.SelectedIndices.Count != 1)
return;
Rectangle rowBounds = olv.GetItem(olv.SelectedIndices[0]).Bounds;
rowBounds.Inflate(0, 2);
GraphicsPath path = this.GetRoundedRect(rowBounds, 15);
g.DrawPath(new Pen(Color.Red, 2.0f), path);
}
private GraphicsPath GetRoundedRect(RectangleF rect, float diameter) {
GraphicsPath path = new GraphicsPath();
RectangleF arc = new RectangleF(rect.X, rect.Y, diameter, diameter);
path.AddArc(arc, 180, 90);
arc.X = rect.Right - diameter;
path.AddArc(arc, 270, 90);
arc.Y = rect.Bottom - diameter;
path.AddArc(arc, 0, 90);
arc.X = rect.Left;
path.AddArc(arc, 90, 90);
path.CloseFigure();
return path;
}
}这给出了类似于下面的内容:

发布于 2009-08-04 17:34:23
我的第一个想法是将ListView控件子类化,将OwnerDraw设置为true,然后自己执行所有绘制,但对于如此小的更改来说,这似乎有点过头了。
然而,在我的网络漫游中,我发现了这个article,它可能会有帮助,因为它与你的情况非常相似,可以让你避免自己画所有的东西。
https://stackoverflow.com/questions/1228492
复制相似问题