如何在运行时修改 PropertyGrid(添加/删除属性和动态类型/枚举)

28
我该如何在运行时以各种方式修改属性网格?我想能够添加和删除属性并添加“动态类型”,我的意思是一种类型,可以使用TypeConverter在属性网格中生成运行时生成的下拉列表。
我实际上已经能够做到这两件事(添加/删除属性和添加动态类型),但只是分开而不是同时进行。
为了实现在运行时添加和删除属性的支持,我使用了这篇CodeProject文章并稍微修改了代码以支持不同类型(不仅仅是字符串)。
private System.Windows.Forms.PropertyGrid propertyGrid1;
private CustomClass myProperties = new CustomClass();

public Form1()
{
    InitializeComponent();

    myProperties.Add(new CustomProperty("Name", "Sven", typeof(string), false, true));
    myProperties.Add(new CustomProperty("MyBool", "True", typeof(bool), false, true));
    myProperties.Add(new CustomProperty("CaptionPosition", "Top", typeof(CaptionPosition), false, true));
    myProperties.Add(new CustomProperty("Custom", "", typeof(StatesList), false, true)); //<-- doesn't work
}

/// <summary>
/// CustomClass (Which is binding to property grid)
/// </summary>
public class CustomClass: CollectionBase,ICustomTypeDescriptor
{
    /// <summary>
    /// Add CustomProperty to Collectionbase List
    /// </summary>
    /// <param name="Value"></param>
    public void Add(CustomProperty Value)
    {
        base.List.Add(Value);
    }

    /// <summary>
    /// Remove item from List
    /// </summary>
    /// <param name="Name"></param>
    public void Remove(string Name)
    {
        foreach(CustomProperty prop in base.List)
        {
            if(prop.Name == Name)
            {
                base.List.Remove(prop);
                return;
            }
        }
    }

等等...

public enum CaptionPosition
{
    Top,
    Left
}

我的完整解决方案可以在这里下载。

当我添加字符串、布尔值或枚举时,它可以正常工作,但是当我尝试添加"动态类型",例如StatesList时,它就无法工作了。有人知道为什么并能帮助我解决吗?

public class StatesList : System.ComponentModel.StringConverter
{
    private string[] _States = { "Alabama", "Alaska", "Arizona", "Arkansas" };

    public override System.ComponentModel.TypeConverter.StandardValuesCollection
    GetStandardValues(ITypeDescriptorContext context)
    {
        return new StandardValuesCollection(_States);
    }

    public override bool GetStandardValuesSupported(ITypeDescriptorContext context)
    {
        return true;
    }

    public override bool GetStandardValuesExclusive(ITypeDescriptorContext context)
    {
        return true;
    }
}

当您不尝试在运行时添加属性时,使用TypeConverter的方法可以正常工作,例如this code可以无任何问题地运行,但我希望能够同时实现两者。
请查看我的项目。 谢谢!
4个回答

8
你需要做的是添加一个StatesList(TypeConverter)属性。
你应该添加一个以StatesList为其TypeConverter的属性。

6
啊,当然!
myProperties.Add(new CustomProperty("Custom", "", typeof(States), false, true));

[TypeConverter(typeof(StatesList))]
public class States
{
}

太棒了,非常感谢!

我已经更新了我的项目,希望对其他人有所帮助。你可以在这里找到它。


我知道这篇文章很老了,但我有一个简短的问题。我在我的应用程序中使用了您的代码,但自定义组合框不起作用。属性名称被灰显,并且没有可用选项。您有什么想法可以解决这个问题吗?提前致谢, 马丁 - user653427
1
@user653427,由于这是一个非常老的问题,因此发布一个新问题并链接回这个页面可能也值得一试。这样你就可以提供更多细节并吸引更多人了。 - Bill the Lizard
为了避免额外的空“States”类,您还可以在PropertyDescriptor实现(CustomPropertyDescriptor)中返回typeconverter - 在您的情况下要覆盖的成员是:public TypeConverter Converter。 - Chanakya
谢谢,如果用户从CaptionPosition组合框中选择了顶部或底部,我们如何在代码中获取该值? - Nastaran Hakimi
另外一个问题是当你为CaptionPosition枚举类型新建CustomProperty时,你将"top"设置为其值。然而,在我运行应用程序时,一开始captionPosition字段中并没有"top"值。只有在我从comboBox中选择项后才显示相关的值。我该如何解决这个问题?我的意思是如何为combobox定义默认值。 - Nastaran Hakimi

5
这个问题和答案对我非常有用。但是,我需要进一步扩展它,允许在运行时生成下拉列表值。如果有人觉得有用的话,我想发布一些示例代码。
首先,我向CustomProperty构造函数添加了一个选项参数,并添加了Options属性:
    private List<string> lOptions;

    public CustomProperty(string sName, object value, Type tType, bool bReadOnly, bool bVisible, List<string> lOptions)
    {
        this.lOptions = lOptions;
    }

    public List<string> Options
    {
        get { return lOptions; }
    }

其次,我在CustomPropertyDescriptor类中添加了一个Options属性:

    public List<string> Options
    {
        get
        {
            return m_Property.Options;
        }
    }

第三步,我需要修改我的动态类型类(即StatesList)中的GetStandardValues方法,以利用CustomPropertyDescriptor对象上的新Options属性:

    public override StandardValuesCollection
                 GetStandardValues(ITypeDescriptorContext context)
    {
        CustomPropertyDescriptor descriptor = (CustomPropertyDescriptor)context.PropertyDescriptor;
        return new StandardValuesCollection(descriptor.Options);
    }

最后,当创建一个新的CustomProperty对象时,我必须传入我的选项列表:
    List<string> optionsList = new List<string>(new string[] { "test1", "test2", "test3" });        
    CustomProperty myProperty = new CustomProperty(attr.Name, attr.Value, valueType, false, true, optionsList);

在此示例中,我传递的是静态列表,但您可以以任何方式生成下拉菜单的选项列表,从而完全控制可用选项。


感谢您抽出时间对答案进行详细解释,非常有帮助。 - Casper Leon Nielsen
valueType是什么?它的typeof(list<string>)是什么? - Nastaran Hakimi
传递的属性值的类型可能是字符串、整数或自定义类型,例如 States(请参见 @salle55 的答案)。您可以使用 typeof,就像您指示的那样,或者可能使用 value.GetType()。 - Scott

0
在我的情况下,TypeConverter 没有应用于 States 类。
[TypeConverter(typeof(StatesList))] // not work
public class States
{
}

所以我在CustomPropertyDescriptor中添加了override

public override TypeConverter Converter
{
    get {
        if (this.PropertyType.Equals(typeof(States)) ) {
            return new StatesList(); ; 
        }
        return base.Converter;
    }
}

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