C#/winforms: 如何最好地绑定一个 PropertyGrid 和一个 System.Data.DataRow?

8

我有一些包含多个字段的System.Data.DataRows,其中大部分只是普通类型,如int、single和string。

使用propertygrid使它们可编辑的最佳方法是什么? 它应该自动工作,无论DataRows具有哪种类型的字段,但不应显示所有字段。我想提供一个应该隐藏的属性列表。

由于DataTable是自动生成的,我无法添加自定义属性,如[Browsable(false)]。

非常感谢!


1
更改示例;现在适用于筛选列。 - Marc Gravell
1个回答

17

修改以处理过滤; 很棘手:除了获取 DataRowView 之外,我们需要提供一个自定义组件,通过传递 PropetyDescriptor 来假装(伪装成)DataRowView(它本身正在伪装成 DataRow),并过滤掉我们不想要的属性。

非常有趣的问题 ;-p 在经典类中更容易解决,但以下代码适用于 DataRow ;-p

请注意,您可以在这个区域做其他事情,使一些属性不可编辑(IsReadOnly),或者具有不同的标题(DisplayName),或者类别(Category) - 通过覆盖 RowWrapperDescriptor 中的其他成员。

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Windows.Forms;
static class program
{
    [STAThread]
    static void Main()
    {
        DataTable table = new DataTable();
        table.Columns.Add("ID", typeof(int));
        table.Columns.Add("Foo", typeof(int));
        table.Columns.Add("Bar", typeof(string));
        table.Columns.Add("Audit", typeof(DateTime));

        table.Rows.Add(1, 14, "abc", DateTime.MinValue);
        DataRow row = table.Rows.Add(2,13,"def", DateTime.MinValue);
        table.Rows.Add(3, 24, "ghi", DateTime.MinValue);

        RowWrapper wrapper = new RowWrapper(row);
        wrapper.Exclude.Add("ID");
        wrapper.Exclude.Add("Bar");

        Application.EnableVisualStyles();
        Application.Run(new Form {Controls = {
            new PropertyGrid { Dock = DockStyle.Fill,
                SelectedObject = wrapper}}});
    }
}

[TypeConverter(typeof(RowWrapper.RowWrapperConverter))]
class RowWrapper
{
    private readonly List<string> exclude = new List<string>();
    public List<string> Exclude { get { return exclude; } }
    private readonly DataRowView rowView;
    public RowWrapper(DataRow row)
    {
        DataView view = new DataView(row.Table);
        foreach (DataRowView tmp in view)
        {
            if (tmp.Row == row)
            {
                rowView = tmp;
                break;
            }
        }
    }
    static DataRowView GetRowView(object component)
    {
        return ((RowWrapper)component).rowView;
    }
    class RowWrapperConverter : TypeConverter
    {
        public override bool GetPropertiesSupported(ITypeDescriptorContext context)
        {
            return true;
        }
        public override PropertyDescriptorCollection GetProperties(
            ITypeDescriptorContext context, object value, Attribute[] attributes)
        {
            RowWrapper rw = (RowWrapper)value;
            PropertyDescriptorCollection props = TypeDescriptor.GetProperties(
                GetRowView(value), attributes);
            List<PropertyDescriptor> result = new List<PropertyDescriptor>(props.Count);
            foreach (PropertyDescriptor prop in props)
            {
                if (rw.Exclude.Contains(prop.Name)) continue;
                result.Add(new RowWrapperDescriptor(prop));
            }
            return new PropertyDescriptorCollection(result.ToArray());
        }
    }
    class RowWrapperDescriptor : PropertyDescriptor
    {
        static Attribute[] GetAttribs(AttributeCollection value)
        {
            if (value == null) return null;
            Attribute[] result = new Attribute[value.Count];
            value.CopyTo(result, 0);
            return result;
        }
        readonly PropertyDescriptor innerProp;
        public RowWrapperDescriptor(PropertyDescriptor innerProperty)
            : base(
                innerProperty.Name, GetAttribs(innerProperty.Attributes))
        {
            this.innerProp = innerProperty;
        }


        public override bool ShouldSerializeValue(object component)
        {
            return innerProp.ShouldSerializeValue(GetRowView(component));
        }
        public override void ResetValue(object component)
        {
            innerProp.ResetValue(GetRowView(component));
        }
        public override bool CanResetValue(object component)
        {
            return innerProp.CanResetValue(GetRowView(component));
        }
        public override void SetValue(object component, object value)
        {
            innerProp.SetValue(GetRowView(component), value);
        }
        public override object GetValue(object component)
        {
            return innerProp.GetValue(GetRowView(component));
        }
        public override Type PropertyType
        {
            get { return innerProp.PropertyType; }
        }
        public override Type ComponentType
        {
            get { return typeof(RowWrapper); }
        }
        public override bool IsReadOnly
        {
            get { return innerProp.IsReadOnly; }
        }
    }
}

好的,谢谢。但是我该如何在这个视图中隐藏某些属性呢?例如,我不希望表格的“Id”被更改。 - clamp
顺便说一句,由于DataTable是自动生成的,我无法添加自定义属性,如[Browsable(false)]。谢谢! - clamp
1
太棒了,可惜我不能给它投两票。 - user256890

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接