DataGridView Using SortableBindingList
Asked Answered
S

2

15

I have a function that returns an IList< T > and is the DataSource for a DataGridView. I learned that DataGridView won't sort IList. I read This stackoverflow Q&A and am trying to implement SortableBindingList. I must be doing something wrong because my DataGridView is empty. I also tried to access an element from the SortableBindingSource with a TextBox and nothing as well.

using Microsoft.SqlServer.Management.Controls;
public partial class Form1 : Form
{
    IBusinessLayer businessLayer;
    IList<Category> categories;
    SortableBindingList<Category> catSortable;

    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {

        businessLayer = new BusinessLayer();

        categories = businessLayer.GetAllCategories();
        catSortable = new SortableBindingList<Category>(categories);
        categoryBindingSource.DataSource = catSortable;
        categoryDataGridView.DataSource = categoryBindingSource;

        textBox1.Text = catSortable[0].CategoryName;

    }
}

I inspected the Microsoft.SqlServer.Management.Controls, does this look right?

namespace Microsoft.SqlServer.Management.Controls
{
    public class SortableBindingList<T> : BindingList<T>
    {
        public SortableBindingList();
        public SortableBindingList(IList<T> list);

        protected override bool IsSortedCore { get; }
        protected override ListSortDirection SortDirectionCore { get; }
        protected override PropertyDescriptor SortPropertyCore { get; }
        protected override bool SupportsSortingCore { get; }

        protected override void ApplySortCore(PropertyDescriptor prop, ListSortDirection direction);
        protected override void RemoveSortCore();
    }
}

I really appreciate the help and helping me learn. Thanks everyone!

Stutter answered 14/5, 2014 at 17:22 Comment(1)
I got this to work by creating my own SortableBindingList class like in the stackoverflow example. I wanted to use the Microsoft.SqlServer.Management.Controls.dll though. Whats the deal?Stutter
E
33

Try this SortableBindingList:

public class SortableBindingList<T> : BindingList<T>
{
    private bool isSortedValue;
    ListSortDirection sortDirectionValue;
    PropertyDescriptor sortPropertyValue;

    public SortableBindingList()
    {
    }

    public SortableBindingList(IList<T> list)
    {
        foreach (object o in list)
        {
            this.Add((T)o);
        }
    }

    protected override void ApplySortCore(PropertyDescriptor prop,
        ListSortDirection direction)
    {
        Type interfaceType = prop.PropertyType.GetInterface("IComparable");

        if (interfaceType == null && prop.PropertyType.IsValueType)
        {
            Type underlyingType = Nullable.GetUnderlyingType(prop.PropertyType);

            if (underlyingType != null)
            {
                interfaceType = underlyingType.GetInterface("IComparable");
            }
        }

        if (interfaceType != null)
        {
            sortPropertyValue = prop;
            sortDirectionValue = direction;

            IEnumerable<T> query = base.Items;

            if (direction == ListSortDirection.Ascending)
            {
                query = query.OrderBy(i => prop.GetValue(i));
            }
            else
            {
                query = query.OrderByDescending(i => prop.GetValue(i));
            }

            int newIndex = 0;
            foreach (object item in query)
            {
                this.Items[newIndex] = (T)item;
                newIndex++;
            }

            isSortedValue = true;
            this.OnListChanged(new ListChangedEventArgs(ListChangedType.Reset, -1));
        }
        else
        {
            throw new NotSupportedException("Cannot sort by " + prop.Name +
                ". This" + prop.PropertyType.ToString() +
                " does not implement IComparable");
        }
    }

    protected override PropertyDescriptor SortPropertyCore
    {
        get { return sortPropertyValue; }
    }

    protected override ListSortDirection SortDirectionCore
    {
        get { return sortDirectionValue; }
    }

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

    protected override bool IsSortedCore
    {
        get { return isSortedValue; }
    }
}
Extranuclear answered 14/5, 2014 at 17:31 Comment(5)
Very similar to the one I am using now. Yours seems a bit more "slim" and throws NotSupportedException. Thanks for the quick reply! Why is the Microsoft.SqlServer.Management.Controls.dll from C:\Program Files\Microsoft SQL Server\100\Setup Bootstrap\Release\x64 empty and not workStutter
@Stutter I wanted to try out how it works for me, but now I'm having issues with compatibility issues of my project architecture and processor arhitecture when I import Microsoft.SqlServer.Management.Controls.dll and it simply won't run :)Extranuclear
I get that warning as well. I will install the 32bit version of SQL Server or steal the Microsoft.SqlServer.Management.Controls.dll from one of my servers and see if that makes a difference. Thanks!Stutter
This SortableBindingList fixed a problem with a different implementation I had been using. That one wouldn't allow me to add items. I did notice the ArrayList var doesn't appear to be used.Unavailing
If your query implements IEnumerable<T>, why do you use foreach(object item in query) instead of foreach(T item in query)?Moshemoshell
W
0

Try this, don`t forget to set SortMode to columns. This SortableBindingList retains the original order of elements when reopened.

public class SortableBindingList<T> : BindingList<T>
{
    // reference to the list provided at the time of instantiation
    private List<T> _originalList;
    private ListSortDirection? _sortDirection;
    private PropertyDescriptor _sortProperty;

    protected override bool IsSortedCore => _sortDirection != null;
    protected override bool SupportsSortingCore => true;
    protected override ListSortDirection SortDirectionCore => _sortDirection ?? ListSortDirection.Ascending;
    protected override PropertyDescriptor SortPropertyCore => _sortProperty;

    public SortableBindingList()
    {
        _originalList = new List<T>();
    }

    public SortableBindingList(List<T> list)
    {
        _originalList = list;
        ResetItems(_originalList);
    }

    public void Sort(PropertyDescriptor property, ListSortDirection direction) => this.ApplySortCore(property, direction);

    protected override void ApplySortCore(PropertyDescriptor property, ListSortDirection direction)
    {
        _sortProperty = property;
        _sortDirection = direction;

        if (!(Items is List<T> items))
            return;

        items.Sort((left, right) =>
        {
            var compareRes = Compare(left == null ? null : _sortProperty.GetValue(left),
                                        right == null ? null : _sortProperty.GetValue(right));
            if (_sortDirection == ListSortDirection.Descending)
                return -compareRes;
            return compareRes;
        });

        ResetBindings();
    }

    private static int Compare(object lhs, object rhs)
    {
        if (lhs == null)
            return rhs == null ? 0 : -1;
        if (rhs == null)
            return 1;
        if (lhs is IComparable comparable)
            return comparable.CompareTo(rhs);
        return lhs.Equals(rhs) ? 0 : string.Compare(lhs.ToString(), rhs.ToString(), StringComparison.Ordinal);
    }

    protected override void RemoveSortCore()
    {
        _sortDirection = null;
        ResetItems(_originalList);
    }

    private void ResetItems(List<T> items)
    {
        base.ClearItems();

        for (var i = 0; i < items.Count; i++)
        {
            base.InsertItem(i, items[i]);
        }
    }

    protected override void OnListChanged(ListChangedEventArgs e)
    {
        _originalList = Items.ToList();
    }
}
Wald answered 14/10, 2021 at 7:23 Comment(1)
I know it is an old post here, but is there a way to add a GetRange method to the SortableBindingList class?Schaal

© 2022 - 2024 — McMap. All rights reserved.