Reputation: 10402
Is it possible to ownerdraw the entire column header section of a listview? (including the region to the right of the column headers)? ListView is in Details View.
An answer here indicates that the remaining space can be drawn along with the last column header: http://www.devnewsgroups.net/group/microsoft.public.dotnet.framework.windowsforms/topic32927.aspx
But it does not seem to work at all - nothing is drawn outside header area.
The proposed solution is based on drawing outside of the passed Bounds:
if (e.ColumnIndex == 3) //last column index
{
Rectangle rc = new Rectangle(e.Bounds.Right, //Right instead of Left - offsets the rectangle
e.Bounds.Top,
e.Bounds.Width,
e.Bounds.Height);
e.Graphics.FillRectangle(Brushes.Red, rc);
}
The ClipBounds property of the available Graphics instance indicates an unbound area (from large negative numbers to large positive). But nothing is drawn outside the columnheader area of the last column.
Does anybody have a solution for this?
Upvotes: 8
Views: 6688
Reputation: 236
Grammarian answer was so close, it helped me get something working that is almost perfect
his 1. Hi tech and partially effective was on the right track.
but instead of drawing in the DrawColumnHeader override the listview in your own class so you can easily add a wndproc override looking for the WM_PAINT message.
I guess he figured that out and that's why he wrote the ObjectListView control (link seems to be broken)
it also involved catching the ColumnWidthChanged event to invalidate the control as the wndproc does not fire for that by itself.
the only small things I can fault with this is
but the rest of the time it draws pretty perfect, for the code size compared to other more complex HeaderControl type answers
public class ListViewColor : ListView
{
[System.Runtime.InteropServices.DllImport("user32")]
private static extern IntPtr GetDC(IntPtr hwnd);
[System.Runtime.InteropServices.DllImport("user32")]
private static extern IntPtr ReleaseDC(IntPtr hwnd, IntPtr hdc);
[System.Runtime.InteropServices.DllImport("user32.dll")]
public static extern IntPtr SendMessage(IntPtr hWnd, uint wMsg, IntPtr wParam, IntPtr lParam);
public static IntPtr GetHeaderControl(ListView list)
{
const int LVM_GETHEADER = 0x1000 + 31;
return SendMessage(list.Handle, LVM_GETHEADER, 0, 0);
}
public ListViewColor()
{
if (this.DesignMode == false)
{
this.DrawColumnHeader += lvw_DrawColumnHeader;
this.DrawItem += lvw_DrawItem;
this.DrawSubItem += lvw_DrawSubItem;
this.ColumnWidthChanged += lvw_ColumnWidthChanged;
this.OwnerDraw = true;
}
}
protected override void Dispose(bool disposing)
{
this.DrawColumnHeader -= lvw_DrawColumnHeader;
this.DrawItem -= lvw_DrawItem;
this.DrawSubItem -= lvw_DrawSubItem;
this.ColumnWidthChanged -= lvw_ColumnWidthChanged;
base.Dispose(disposing);
}
private void lvw_DrawColumnHeader(object sender, DrawListViewColumnHeaderEventArgs e)
{
// Set the background color
e.Graphics.FillRectangle(Brushes.Gray, e.Bounds);
// Draw the text
e.Graphics.DrawString(e.Header.Text, e.Font, Brushes.White, new Rectangle(e.Bounds.Left, e.Bounds.Top + 1, e.Bounds.Width, e.Bounds.Height));
// Draw the border
e.DrawDefault = false;
}
private void lvw_DrawItem(object sender, DrawListViewItemEventArgs e)
{
e.DrawDefault = true;
}
private void lvw_DrawSubItem(object sender, DrawListViewSubItemEventArgs e)
{
e.DrawDefault = true;
}
private void lvw_ColumnWidthChanged(object sender, ColumnWidthChangedEventArgs e)
{
this.Invalidate();
}
protected override void WndProc(ref Message m)
{
base.WndProc(ref m);
if (m.Msg == 0x0F) // WM_PAINT
{
int totalColumnWidth = 0;
int intHeight = 0;
foreach (ColumnHeader column in this.Columns)
{
totalColumnWidth += column.Width;
}
IntPtr headerControl = GetHeaderControl(this);
IntPtr hdc = GetDC(headerControl);
using (Graphics g = Graphics.FromHdc(hdc))
{
Rectangle rect = new Rectangle(totalColumnWidth, 0, this.Width - totalColumnWidth, this.Font.Height + 4);
g.FillRectangle(Brushes.Gray, rect);
}
ReleaseDC(headerControl, hdc);
}
}
}
Upvotes: 0
Reputation: 993
I went with @grammarian's number 2 as didn't want want to mess around with InteropServices. This solution just uses standard .net. As above, put a spare column at the end with nothing in the Text property (in the vid below, I use "{filler}" just to help with seeing what's going on). Then use various event handlers to work their magic. The result has a few rough edges, but I think it's very passable. Vid of it in action:
And relevant code:
Dim ResizingFillerColumn As Boolean = False
Private Sub ListView_Resize(sender As Object, e As EventArgs) Handles ListView.Resize
ResizeFillerColumn()
End Sub
Private Sub ResizeFillerColumn()
Dim columnsWidth = 0
For i = 0 To ListView.Columns.Count - 2
columnsWidth += ListView.Columns(i).Width
Next
ResizingFillerColumn = True
ListView.Columns(ListView.Columns.Count - 1).Width = ListView.Width - columnsWidth
ResizingFillerColumn = False
End Sub
Private Sub ListView_ColumnReordered(sender As Object, e As ColumnReorderedEventArgs) Handles ListView.ColumnReordered
Dim FillerColumnIndex = ListView.Columns.Count - 1
If e.OldDisplayIndex = FillerColumnIndex Then e.Cancel = True
If e.NewDisplayIndex = FillerColumnIndex Then e.Cancel = True
End Sub
Private Sub ListView_ColumnWidthChanged(sender As Object, e As ColumnWidthChangedEventArgs) Handles ListView.ColumnWidthChanged
If ResizingFillerColumn Then Return
ResizeFillerColumn()
End Sub
Private Sub ListView_ColumnWidthChanging(sender As Object, e As ColumnWidthChangingEventArgs) Handles ListView.ColumnWidthChanging
If ResizingFillerColumn Then Return
ResizeFillerColumn()
End Sub
Upvotes: 0
Reputation: 6882
I'm surprised by Jeffery Tan's answer in that post. His solution cannot work, since the code tries to draw outside of the header control client area. The hDC
used within custom drawing (and hence owner drawing) is for the client area of the control, and so cannot be used to paint in the non-client area. The area to the right of the right most column in a header control is in non-client area. So you need a different solution.
Possible Solutions
You can enable drawing outside the client area by using the GetDC()
WinAPI call:
[System.Runtime.InteropServices.DllImport("user32")]
private static extern IntPtr GetDC(IntPtr hwnd);
[System.Runtime.InteropServices.DllImport("user32")]
private static extern IntPtr ReleaseDC(IntPtr hwnd, IntPtr hdc);
public static IntPtr GetHeaderControl(ListView list) {
const int LVM_GETHEADER = 0x1000 + 31;
return SendMessage(list.Handle, LVM_GETHEADER, 0, 0);
}
In your column draw event handler, you will need something like this:
if (e.ColumnIndex == 3) //last column index
{
ListView lv = e.Header.ListView;
IntPtr headerControl = NativeMethods.GetHeaderControl(lv);
IntPtr hdc = GetDC(headerControl);
Graphics g = Graphics.FromHdc(hdc);
// Do your extra drawing here
Rectangle rc = new Rectangle(e.Bounds.Right, //Right instead of Left - offsets the rectangle
e.Bounds.Top,
e.Bounds.Width,
e.Bounds.Height);
e.Graphics.FillRectangle(Brushes.Red, rc);
g.Dispose();
ReleaseDC(headerControl, hdc);
}
But the problem with this is that since your drawing is outside the client area, Windows doesn't always know when it should be drawn. So it will disappear sometimes, and then be redrawn when Windows thinks the header needs repainting.
Add an extra empty column to your control, owner draw it do look however you want, make it very wide, and turn off horizontal scrolling (optional).
I know this is horrible, but you're looking for suggestions :)
Use ObjectListView. This wrapper around a .NET ListView allows you to add overlays to your list -- an overlay can draw anywhere within the ListView, including the header. [Declaration: I'm the author of ObjectListView, but I still think it is best solution]
public class HeaderOverlay : AbstractOverlay
{
public override void Draw(ObjectListView olv, Graphics g, Rectangle r) {
if (olv.View != System.Windows.Forms.View.Details)
return;
Point sides = NativeMethods.GetColumnSides(olv, olv.Columns.Count-1);
if (sides.X == -1)
return;
RectangleF headerBounds = new RectangleF(sides.Y, 0, r.Right - sides.Y, 20);
g.FillRectangle(Brushes.Red, headerBounds);
StringFormat sf = new StringFormat();
sf.Alignment = StringAlignment.Center;
sf.LineAlignment = StringAlignment.Center;
g.DrawString("In non-client area!", new Font("Tahoma", 9), Brushes.Black, headerBounds, sf);
}
}
[Reading over this answer, I think this is an example of trying too hard :) Hope you find something here helpful.]
Upvotes: 8