gpt4 book ai didi

c# - 单击列标题时,如何对 DataGridView 中的 DataBound 列进行排序?

转载 作者:行者123 更新时间:2023-11-30 14:39:39 24 4
gpt4 key购买 nike

我相信这应该是自动处理的。我将 DataGridView 绑定(bind)到一个对象数组:

public class Entity {    
public string Name { get; set; }
public int PrimaryKey { get; set; }
}

绑定(bind)网格:

public void BindGrid(Entity[] entities) {
grdEntities.DataSource = entities;
}

当我单击“名称”列中的列标题时,没有任何反应,即使 SortMode 设置为 Automatic。列标题中也缺少排序字形。

我已经尝试绑定(bind)到 IBindingListIList 但这没有用。

我希望有一个简单、优雅的解决方案,可以在 DataGridViewDataGridViewColumn 上设置属性,而不必创建一个新类来支持排序。我应该如何通过单击 DataBound DataGridView 上的标题来支持对列进行排序?

最佳答案

我创建了一个新的基于 IComparer 的接口(interface),允许您指定列和方向。我这样做只是因为我需要我的排序代码尽可能通用——我有两个需要像这样排序的网格,我不想维护两倍的代码。这是界面,非常简单:

   public interface IByColumnComparer : IComparer
{
string SortColumn { get; set; }
bool SortDescending { get; set; }
}

显然,如果您不担心保持通用性(您可能应该这样做),那么这并不是绝对必要的。然后,我构建了一个基于 BindingList<> 的新类。这允许我覆盖排序代码并逐列提供我自己的 IByColumnComparer,这是我需要的灵 active 。检查一下:

public class SortableGenericCollection<T> : BindingList<T>
{
IByColumnComparer GenericComparer = null;

public SortableGenericCollection(IByColumnComparer SortingComparer)
{
GenericComparer = SortingComparer;
}


protected override bool SupportsSortingCore
{
get
{
return true;
}
}

protected override bool IsSortedCore
{
get
{
for (int i = 0; i < Items.Count - 1; ++i)
{
T lhs = Items[i];
T rhs = Items[i + 1];
PropertyDescriptor property = SortPropertyCore;
if (property != null)
{
object lhsValue = lhs == null ? null :
property.GetValue(lhs);
object rhsValue = rhs == null ? null :
property.GetValue(rhs);
int result;
if (lhsValue == null)
{
result = -1;
}
else if (rhsValue == null)
{
result = 1;
}
else
{
result = GenericComparer.Compare(lhs, rhs);
}
if (result >= 0)
{
return false;
}
}
}
return true;
}
}

private ListSortDirection sortDirection;
protected override ListSortDirection SortDirectionCore
{
get
{
return sortDirection;
}
}

private PropertyDescriptor sortProperty;
protected override PropertyDescriptor SortPropertyCore
{
get
{
return sortProperty;
}
}

protected override void ApplySortCore(PropertyDescriptor prop,
ListSortDirection direction)
{
sortProperty = prop;
sortDirection = direction;

GenericComparer.SortColumn = prop.Name;
GenericComparer.SortDescending = direction == ListSortDirection.Descending ? true : false;

List<T> list = (List<T>)Items;
list.Sort(delegate(T lhs, T rhs)
{
if (sortProperty != null)
{
object lhsValue = lhs == null ? null :
sortProperty.GetValue(lhs);
object rhsValue = rhs == null ? null :
sortProperty.GetValue(rhs);
int result;
if (lhsValue == null)
{
result = -1;
}
else if (rhsValue == null)
{
result = 1;
}
else
{
result = GenericComparer.Compare(lhs, rhs);
}
return result;
}
else
{
return 0;
}
});
}

protected override void RemoveSortCore()
{
sortDirection = ListSortDirection.Ascending;
sortProperty = null;
}
}

编辑 这应该提供一些关于如何根据我上面的界面创建您自己的 IComparer 的信息。 拥有自己的基于接口(interface)的 IComparer 的优势在于,您可以对一些列进行排序,而对其他列进行另一种排序(一些列可能是字符串,一些是整数,有些可能对顶部的内容有特殊的规则,等等)。以下是您的 IComparer 可能如何工作的示例:

public class MyGenericComparer : IByColumnComparer
{
private string columnToCompare;
private bool descending;

public string SortColumn
{
get { return columnToCompare; }
set { columnToCompare = value; }
}

public bool SortDescending
{
get { return descending; }
set { descending = value; }
}

public MyGenericComparer(string column, bool descend)
{
columnToCompare = column;
descending = descend;
}

public int Compare(object x, object y)
{
MyGenericObject firstObj = (MyGenericObject )x;
MyGenericObject secondObj = (MyGenericObject )y;

if (descending)
{
MyGenericObject tmp = secondObj ;
secondObj = firstObj ;
firstObj = tmp;
}

if (columnToCompare == "StringColumn")
{
//Run code to compare strings, return the appropriate int
//eg, "1" if firstObj was greater, "-1" is secondObj, "0" if equal
}

if (columnToCompare == "IntColumn")
{
//Run code to compare ints, return the appropriate int
//eg, "1" if firstObj was greater, "-1" is secondObj, "0" if equal
}
}
}

然后您所要做的就是使用您的比较器实例创建您的列表!

public static MyGenericComparer GridComparer = new MyGenericComparer();
public static SortableGenericCollection<GenericObject> GridList = new SortableGenericCollection<GenericObject>(GridComparer);

关于c# - 单击列标题时,如何对 DataGridView 中的 DataBound 列进行排序?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6228340/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com