c# - WinForms DataGridView - 数据绑定(bind)到具有列表属性(列数可变)的对象

标签 c# .net winforms data-binding datagridview

我有一个 .NET 类,我想在 DataGridView 中显示,默认数据绑定(bind)——将 DGV 的数据源设置为对象——产生了我 90% 的要求(即它正确输出公共(public)属性,我可以添加轻松排序)。

但是,我需要绑定(bind)的属性之一是一个列表,其中包含需要位于其他数据绑定(bind)项之后的单独列中的数据。我一直在思考如何最好地实现这一点。

我的类(class)看起来像这样:

public class BookDetails
{
    public string Title { get; set; }
    public int TotalRating { get; set; }
    public int Occurrence { get; set; }
    public List<int> Rating { get; set; }
}

理想情况下,我能够将该 Rating 属性扩展为多个数字列,以便在运行时提供如下输出:

标题 |总评分 |发生 | R1 | R2 | R3 ... RN

将总评分计算为所有单个评分的总和也很有用,但我目前正在手动更新它,没有问题。

最佳答案

像这样?

using System;
using System.Collections;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Windows.Forms;

public class BookDetails
{
    public string Title { get; set; }
    public int TotalRating { get; set; }
    public int Occurrence { get; set; }
    public List<int> Rating { get; set; }
}

class BookList : List<BookDetails>, ITypedList
{

    public PropertyDescriptorCollection GetItemProperties(PropertyDescriptor[] listAccessors)
    {
        var origProps = TypeDescriptor.GetProperties(typeof(BookDetails));
        List<PropertyDescriptor> newProps = new List<PropertyDescriptor>(origProps.Count);
        PropertyDescriptor doThisLast = null;
        foreach (PropertyDescriptor prop in origProps)
        {

            if (prop.Name == "Rating") doThisLast = prop;
            else newProps.Add(prop);
        }
        if (doThisLast != null)
        {
            var max = (from book in this
                       let rating = book.Rating
                       where rating != null
                       select (int?)rating.Count).Max() ?? 0;
            if (max > 0)
            {
                // want it nullable to account for jagged arrays
                Type propType = typeof(int?); // could also figure this out from List<T> in
                                              // the general case, but make it nullable
                for (int i = 0; i < max; i++)
                {
                    newProps.Add(new ListItemDescriptor(doThisLast, i, propType));
                }
            }
        }
        return new PropertyDescriptorCollection(newProps.ToArray());
    }

    public string GetListName(PropertyDescriptor[] listAccessors)
    {
        return "";
    }
}

class ListItemDescriptor : PropertyDescriptor
{
    private static readonly Attribute[] nix = new Attribute[0];
    private readonly PropertyDescriptor tail;
    private readonly Type type;
    private readonly int index;
    public ListItemDescriptor(PropertyDescriptor tail, int index, Type type) : base(tail.Name + "[" + index + "]", nix)
    {
        this.tail = tail;
        this.type = type;
        this.index = index;
    }
    public override object GetValue(object component)
    {
        IList list = tail.GetValue(component) as IList;
        return (list == null || list.Count <= index) ? null : list[index];
    }
    public override Type PropertyType
    {
        get { return type; }
    }
    public override bool IsReadOnly
    {
        get { return true; }
    }
    public override void SetValue(object component, object value)
    {
        throw new NotSupportedException();
    }
    public override void ResetValue(object component)
    {
        throw new NotSupportedException();
    }
    public override bool CanResetValue(object component)
    {
        return false;
    }
    public override Type ComponentType
    {
        get { return tail.ComponentType; }
    }
    public override bool ShouldSerializeValue(object component)
    {
        return false;
    }
}

static class Program
{
    [STAThread]
    static void Main()
    {
        Application.EnableVisualStyles();
        var data = new BookList {
            new BookDetails { Title = "abc", TotalRating = 3, Occurrence = 2, Rating = new List<int> {1,2,1}},
            new BookDetails { Title = "def", TotalRating = 3, Occurrence = 2, Rating = null },
            new BookDetails { Title = "ghi", TotalRating = 3, Occurrence = 2, Rating = new List<int> {3, 2}},
            new BookDetails { Title = "jkl", TotalRating = 3, Occurrence = 2, Rating = new List<int>()},
        };
        Application.Run(new Form
        {
            Controls = {
                new DataGridView {
                    Dock = DockStyle.Fill,
                    DataSource = data
                }
            }
        });

    }
}

关于c# - WinForms DataGridView - 数据绑定(bind)到具有列表属性(列数可变)的对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4716092/

相关文章:

c# - FakeItEasy.Invokes() 中调用的方法名称

c# - 在 WPF 中使用 Backgroundworker 弹出主应用程序的 MessageBox

c# - 默认情况下,Windows 工具提示使用什么系统颜色属性作为背景颜色?

c# - 如何在有选择的情况下获取 TextBox 的插入位置?

c# - 如何取消通过数据绑定(bind)所做的更改?

c# - 在 .NET 的控制台窗口中更改字体

c# - 创建虚拟 COM 端口

.net - 为什么 NLog.LogManager.Configuration 返回 null?

c# - 解释 NetCore 上 C# TcpClient/TcpListener 的奇怪行为

c# - 关闭 XDOCUMENT 实例