Introduction
.NET's ListView
control is a nice wrapper around the native List control, but the support for column sorting falls a bit short of what's commonly necessary. Some solutions to this problem require you to derive from a class that implements column sorting however. Since inheritance introduces very tight coupling between components, it is not the appropriate solution in every case. For example, your custom ListView
derives from a class that implements functionality that you need except for column sorting and you cannot change the base class.
ListViewSortManager
provides several useful features:
- Easy to use.
- Case-sensitive (
ListViewTextSort
) and case-insensitive (ListViewTextCaseInsensitiveSort
) text sorting.
- Integer (
ListViewInt32Sort
and ListViewInt64Sort
) and floating-point (ListViewDoubleSort
) sorting.
- Date (
ListViewDateSort
) sorting.
- Extendable for other specialized user-provided sort orders.
- Transparently handles null and empty strings.
- Transparently handles ascending and descending orderings.
- Displays the sort order image in the column headers.
- The user can specify the column and sort order at any time.
- Automatic sorting can be disabled for batch insertion of elements to the list.
Usage
- Add a variable of type
ListViewSortManager
. private ListViewSortManager m_sortMgr;
- In your form's constructor, after the call to
InitializeComponent()
, construct the ListViewSortManager
passing the list and an array of Type
for sorters. There should be one entry in the array for each column in your list. The constructor takes care of hooking into the ListView
's ColumnClick
event. public MainForm()
{
InitializeComponent();
m_sortMgr = new ListViewSortManager(m_list,
new Type[] {
typeof(ListViewTextSort),
typeof(ListViewTextCaseInsensitiveSort),
typeof(ListViewIntegerSort),
typeof(ListViewFloatSort),
typeof(ListViewDateSort)
}
);
}
- Voil�! Column sorting has been added to your list.
Extended Usage
If you need to provide a sort ordering that doesn't match the ones provided, you just have to derive a class from ListViewTextSort
and, override the OnCompare()
method which receives two strings that have to be converted to whatever format that your comparison is based on.
public class ListViewDateSort: ListViewTextSort
{
public ListViewDateSort(int column, bool ascending):
base(column, ascending)
{
}
protected override int OnCompare(string lhs, string rhs)
{
return DateTime.Parse(lhs).CompareTo(DateTime.Parse(rhs));
}
}
To Do
- Investigate alternative, more efficient solutions for sorting that don't require text conversions. Probably, using
ListView.Tag
and adding overridables to ListViewSortManager
.
- As soon as C# implements generics, implement the comparers as templates.
Updates
- Added the
SortEnabled
property to enable/disable automatic sorting of the list's elements. Setting SortEnabled
to false
before adding a large number of elements to the list will speed up the insertion considerably. (Thanks to Jimmy S. for the suggestion.)
- Added support for native header sort arrows when using version 6 of the Common Controls library. (Thanks to Arlen Feldman for the suggestion.)
ShowHeaderIcon()
always left-aligned the column header text regardless of the settings. (Thanks to Jezbo for spotting this).
ShowHeaderIcon()
now locates the sorting arrow icon on the opposite side of the column header text.
Version 1.2 (11/08/02)
- The
ListViewSortManager
's constructor now hooks up to the ColumnClick
event making it much easier to use the control. (Thanks to debaser for the suggestion).
- I forgot to set the transparency for the arrow images. (Thanks to Carlo J. Bos and Cory Smith for spotting this).
- Made
Sort()
public
again and added an overload that allows to select the column and sort order used. Also added Column
and SortOrder
properties (Thanks to Cory Smith for the suggestion).
Version 1.1 (11/07/02)
- Implemented ascending/descending arrow images in column headers. No code changes are necessary to benefit from this.
Version 1.0 (04/22/02)