扩展属性和字符串转换TypeConverter和属性窗口
NET属性窗口最重要的一个特性就是可以显示嵌套的属性这样就提供了比属性类别更加细化和更有逻辑的分类嵌套属性对于类目显示和排序显示都是适用的这样可以让属性列表更加紧凑比如我们用带有子属性X和Y的一个Location属性来代替Top和Left两个属性就更加合理
图嵌套属性
不过如何来决定一个属性可以展开呢?这些不是由属性窗口来决定而是取决于属性自己的类型在NET framework中每一种类型都是和一个TypeConverter联系在一起的比如Boolean和string的TypeConverter就不会允许展开因为让boolean类型含有子属性是没有意义的
在NET framework中TypeConverter实际上是执行了不少的方法在属性窗口中就更多了正像他的名字所说明的那样TypeConverter提供了一种动态的从一种类型改变到另一种类型的标准方式事实上属性窗口只和string打交道所以他就依赖于TypeConverter来进行类型之间的转换(主要是和string类型的转换)TypeConverter同样是可以提供扩展性能以及复杂类型来和属性窗口交互
比如看下面这个Person类
[TypeConverter(typeof(PersonConverter))]
public class Person
{
private string firstName = ;
private string lastName = ;
private intage = ;
public int Age
{
get
{
return age;
}
set
{
age = value;
}
}
public string FirstName
{
get
{
return firstName;
}
set
{
thisfirstName = value;
}
}
public string LastName
{
get
{
return lastName;
}
set
{
thislastName = value;
}
}
}
我们注意到Person类被指定了TypeConverterAttribute特性TypeConverterAttribute特性还指定了这个类的类型转换器(PersonConverter)如果没有指定TypeConverterAttribute特性默认使用TypeConverter类对于一些简单数据类型比如FontPoint等TypeConverter可以很好地工作但如果数据类型比较复杂那么它对类型是转换可能就不是我们希望的那样因此我们有必要从TypeConverter派生自己的类型转换器在这里就是PersonConverter本例中我们首先重载了GetPropertiesSupported和GetProperties方法来决定属性是否可以展开
internal class PersonConverter : TypeConverter
{
public override PropertyDescriptorCollection
GetProperties(ITypeDescriptorContext context
object value
Attribute[] filter)
{
return TypeDescriptorGetProperties(value filter);
}
public override bool GetPropertiesSupported(
ITypeDescriptorContext context)
{
return true;
}
}
在通常情况下直接使用TpyeConverter进行转换已经足够了简单的扩展就是从TypeConverter直接派生你所要的类型转换器更复杂的扩展就需要从ExpandableObjectConverter派生类型转换器了现在我们修改PersonConverter来转换一个Person类并且显示一个字符串
internal class PersonConverter : ExpandableObjectConverter
{
public override bool CanConvertFrom(
ITypeDescriptorContext context Type t)
{
if (t == typeof(string))
{
return true;
}
return baseCanConvertFrom(context t);
}
public override object ConvertFrom(
ITypeDescriptorContext context
CultureInfo info
object value)
{
if (value is string)
{
try
{
string s = (string) value;
// parse the format Last First (Age)
//
int comma = sIndexOf();
if (comma != )
{
// now that we have the comma get
// the last name
string last = sSubstring( comma);
int paren = sLastIndexOf(();
if (paren != && sLastIndexOf()) == sLength )
{
// pick up the first name
string first = sSubstring(comma + paren comma );
// get the age
int age = IntParse(
sSubstring(paren +
sLength paren ));
Person p = new Person();
pAge = age;
pLastName = lastTrim();
pFirstName = firstTrim();
return p;
}
}
}
catch {}
// if we got this far complain that we
// couldnt parse the string
//
throw new ArgumentException(
Can not convert + (string)value +
to type Person);
}
return baseConvertFrom(context info value);
}
public override object ConvertTo(
ITypeDescriptorContext context
CultureInfo culture
object value
Type destType)
{
if (destType == typeof(string) && value is Person)
{
Person p = (Person)value;
// simply build the string as Last First (Age)
return pLastName + +
pFirstName + ( + pAgeToString() + );
}
return baseConvertTo(context culture value destType);
}
}
现在看看我们的Person属性在指定了PersonConverter类型转换器之后既可以展开又可以通过两种方式来操作了直接修改和使用子属性
图 实现展开的TypeConverter
要使用上面的代码我们就生成一个UserControl并且写下如下的代码
private Person p = new Person();
public Person Person
{
get
{
return p;
}
set
{
thisp = value;
}
}