WinForms DataGridView – databind a un objeto con una propiedad de lista (número variable de columnas)

Tengo una clase .NET que me gustaría mostrar en un DataGridView, y el enlace de datos por defecto – configurando DataSource de la DGV para el objeto – produce el 90% de mis requisitos (es decir, está produciendo las propiedades públicas correctamente y puedo agregar la clasificación fácilmente) .

Sin embargo, una de las propiedades que necesito vincular es una lista que contiene datos que deben estar en columnas separadas después de los otros elementos de datos. Estoy atascado sobre la mejor manera de implementar esto.

Mi clase se ve así:

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

Idealmente, podría expandir esa propiedad de Calificación en varias columnas numéricas para obtener un resultado como este en tiempo de ejecución:

Título | Puntuación total | Ocurrencia | R1 | R2 | R3 … RN

También sería útil que la Calificación total se calcule como la sum de todas las calificaciones individuales, pero la actualizo manualmente en este momento sin problemas.

¿Me gusta esto?

 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 Rating { get; set; } } class BookList : List, ITypedList { public PropertyDescriptorCollection GetItemProperties(PropertyDescriptor[] listAccessors) { var origProps = TypeDescriptor.GetProperties(typeof(BookDetails)); List newProps = new List(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 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 {1,2,1}}, new BookDetails { Title = "def", TotalRating = 3, Occurrence = 2, Rating = null }, new BookDetails { Title = "ghi", TotalRating = 3, Occurrence = 2, Rating = new List {3, 2}}, new BookDetails { Title = "jkl", TotalRating = 3, Occurrence = 2, Rating = new List()}, }; Application.Run(new Form { Controls = { new DataGridView { Dock = DockStyle.Fill, DataSource = data } } }); } }