如何在运行时以各种方式修改propertygrid?我希望能够添加和删除属性并添加“动态类型”,我的意思是使用TypeConverter在属性网格中生成运行时生成下拉列表的类型。
我实际上已经能够做这两件事(添加/删除属性并添加动态类型),但不能同时单独使用。
为了实现在运行时添加和删除属性的支持,我使用了this codeproject article并稍微修改了代码以支持不同的类型(不仅仅是字符串)。
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
}
我的完整解决方案可以下载here。
当我添加字符串,bool或枚举时,它工作正常,但当我尝试添加像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可以正常工作,但我希望能够同时执行这两种操作。
请查看my project。 谢谢!
答案 0 :(得分:8)
你做的是将StatesList(一个TypeConverter)添加为属性 你应该做的是添加一个使用StatesList作为其TypeConverter的属性。
答案 1 :(得分:5)
啊,当然!
myProperties.Add(new CustomProperty("Custom", "", typeof(States), false, true));
[TypeConverter(typeof(StatesList))]
public class States
{
}
像魅力一样,谢谢你!
我已更新了我的项目,希望它对其他人有所帮助,可以找到here。
答案 2 :(得分:4)
这个问题和答案对我来说非常有用。但是,我需要通过允许运行时生成的下拉列表值来进一步扩展。我想我会发布一些关于它需要的示例代码,以防有人发现它有用。
首先,我在CustomProperty构造函数中添加了一个options参数,并添加了一个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);
代替我为此示例传入的静态列表,您可以以任何方式生成下拉列表的选项列表,让您完全控制可用选项。
答案 3 :(得分:0)
在我的情况下TypeConverter不适用于States class
[TypeConverter(typeof(StatesList))] // not work
public class States
{
}
所以我在CustomPropertyDescriptor中添加了覆盖
public override TypeConverter Converter
{
get {
if (this.PropertyType.Equals(typeof(States)) ) {
return new StatesList(); ;
}
return base.Converter;
}
}