C#基础系列:开发自己的窗体设计器(PropertyGrid显示中文属性名)

既然是一个窗体设计器,那就应该能够设置控件的属性,设置属性最好的当然是PropertyGrid了,我们仅仅需要使用一个PropertyGrid.SelectedObject = Control就可以搞定,让PropertyGrid显示Control的所有属性。可是这里显示的属性名是英文的。对于我们开发人员来说这无可厚非,我们也乐于接受。并且让PropertyGrid显示中文属性名,这对于我们开发人员的使用来说显得多此一举。可是,对于我这种类型的一个应用工具,英文属性名对于很多客户来说可能就很难懂了。所以应该让PrpertyGrid能够显示中文属性名。

如图:

C#基础系列:开发自己的窗体设计器(PropertyGrid显示中文属性名)

另外,对于这样的一个系统。并不是控件的所有属性都是用户希望的,可能用户希望看到的仅仅是控件高度、控件宽度、控件文本。。等等的属性,但是如果直接将一个控件属性全部显示给用户的话,估计对用户造成的干扰和困惑是很大的。如何解决这个问题呢?其实用户控件开发的时候,如果我们不希望此属性在PropertyGrid中显示,我们可以设置这个控件的Attribute,如:

[Browsable(false)]<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

public int Width

{

get { }

set { }

}

通过使用BrowsableAttribute就可以设置将此属性对PropertyGrid隐藏。

你可能要问到了,对于控件来说,其中的很多属性都是直接继承来的,我们并没有办法控制是否对PropertyGrid隐藏啊?呵呵,对啊,这就是我下面要说的解决方法(当然此方法显得不是很灵活,但是对于这种类型的系统的确相当有用)。

在我的解决方式中,我不直接这样PropertyGrid.SelectedObject = Control,而是把这个Control替换成一个专门为此类型的Control设计的类对象上。比如我对TextBox设计了一个TextBoxProperty,这样我们使用的是PropertyGrid.SelectedObject = TextBoxProperty的一个对象。

下面就是TextBoxProperty的代码:

  1. publicclassTextBoxProperty:PropertyBase
  2. {
  3. privateTextBox_Control;
  4. publicTextBoxProperty()
  5. {
  6. }
  7. publicTextBoxProperty(TextBoxcontrol)
  8. {
  9. this._Control=control;
  10. }
  11. [MyControlAttibute("文本","获取或者设置控件文本","")]
  12. publicstringText
  13. {
  14. get{returnthis._Control.Text;}
  15. set
  16. {
  17. this._Control.Text=value;
  18. }
  19. }
  20. [MyControlAttibute("宽度","获取或者设置控件宽度","")]
  21. publicintWidth
  22. {
  23. get{returnthis._Control.Width;}
  24. set
  25. {
  26. this._Control.Width=(int)value;
  27. }
  28. }
  29. [MyControlAttibute("高度","获取或者设置控件高度","")]
  30. publicintHeight
  31. {
  32. get{returnthis._Control.Height;}
  33. set
  34. {
  35. this._Control.Height=(int)value;
  36. }
  37. }
  38. [MyControlAttibute("上边距","获取或者设置控件上边位置","")]
  39. publicintTop
  40. {
  41. get{returnthis._Control.Top;}
  42. set
  43. {
  44. this._Control.Top=value;
  45. }
  46. }
  47. [MyControlAttibute("左边距","获取或者设置控件左边位置","")]
  48. publicintLeft
  49. {
  50. get{returnthis._Control.Left;}
  51. set
  52. {
  53. this._Control.Left=value;
  54. }
  55. }
  56. [MyControlAttibute("背景色","获取或者设置控件背景颜色","")]
  57. publicColorBackColor
  58. {
  59. get{returnthis._Control.BackColor;}
  60. set
  61. {
  62. this._Control.BackColor=value;
  63. }
  64. }
  65. [MyControlAttibute("前景色","获取或者设置控件的前景颜色","")]
  66. publicColorForeColor
  67. {
  68. get{returnthis._Control.ForeColor;}
  69. set
  70. {
  71. this._Control.ForeColor=value;
  72. }
  73. }
  74. }

你从代码里面已经看出了一些端倪了,在TextBoxProperty中每个要属性都增加了MyControlAttibute,具体Attribute的概念和使用方法您可以参考我的另一篇Blog文《C#基础系列:实现自己的ORM(反射以及AttributeORM中的应用)》。这里对Attribute做了详细的介绍(对初学者有效)。所以我就直接贴出MyControlAttibute的代码好了:

  1. publicclassMyControlAttibute:Attribute
  2. {
  3. privatestring_PropertyName;
  4. privatestring_PropertyDescription;
  5. privateobject_DefaultValue;
  6. publicMyControlAttibute(stringName,stringDescription,objectDefalutValue)
  7. {
  8. this._PropertyName=Name;
  9. this._PropertyDescription=Description;
  10. this._DefaultValue=DefalutValue;
  11. }
  12. publicMyControlAttibute(stringName,stringDescription)
  13. {
  14. this._PropertyName=Name;
  15. this._PropertyDescription=Description;
  16. this._DefaultValue="";
  17. }
  18. publicMyControlAttibute(stringName)
  19. {
  20. this._PropertyName=Name;
  21. this._PropertyDescription="";
  22. this._DefaultValue="";
  23. }
  24. publicstringPropertyName
  25. {
  26. get{returnthis._PropertyName;}
  27. }
  28. publicstringPropertyDescription
  29. {
  30. get{returnthis._PropertyDescription;}
  31. }
  32. publicobjectDefaultValue
  33. {
  34. get{returnthis._DefaultValue;}
  35. }
  36. }

通过上面的两段代码,你已经初步看出了在PropertyGrid中显示中文属性以及仅仅显示我们需要的属性的基本思路。下面就介绍的就是怎么让MyControlAttibute中定义了的中文属性名在PropertyGrid中显示出来。下面这段代码,我仅仅贡献了PropertyStub中这个函数的实现。所以不做过多解释了:

  1. publicoverridestringDisplayName
  2. {
  3. get
  4. {
  5. if(info!=null)
  6. {
  7. MyControlAttibuteuicontrolattibute=(MyControlAttibute)Attribute.GetCustomAttribute(info,typeof(MyControlAttibute));
  8. if(uicontrolattibute!=null)
  9. returnuicontrolattibute.PropertyName;
  10. else
  11. {
  12. returninfo.Name;
  13. }
  14. }
  15. else
  16. return"";
  17. }
  18. }

整个代码如下,里面包含两个类,你直接拷贝出来就可以使用了:

  1. publicdelegatevoidPropertyChanged(objectValue);
  2. ///<summary>
  3. ///主要是实现中文化属性显示
  4. ///</summary>
  5. publicclassPropertyBase:ICustomTypeDescriptor
  6. {
  7. ///<summary>
  8. ///下面这段代码来源于:http://www.bluespace.cn/Html/Csdn/2_47/View_4702219.html
  9. ///</summary>
  10. ///<returns></returns>
  11. #regionICustomTypeDescriptor显式接口定义
  12. AttributeCollectionICustomTypeDescriptor.GetAttributes()
  13. {
  14. returnTypeDescriptor.GetAttributes(this,true);
  15. }
  16. stringICustomTypeDescriptor.GetClassName()
  17. {
  18. returnTypeDescriptor.GetClassName(this,true);
  19. }
  20. stringICustomTypeDescriptor.GetComponentName()
  21. {
  22. returnTypeDescriptor.GetComponentName(this,true);
  23. }
  24. TypeConverterICustomTypeDescriptor.GetConverter()
  25. {
  26. returnTypeDescriptor.GetConverter(this,true);
  27. }
  28. EventDescriptorICustomTypeDescriptor.GetDefaultEvent()
  29. {
  30. returnTypeDescriptor.GetDefaultEvent(this,true);
  31. }
  32. PropertyDescriptorICustomTypeDescriptor.GetDefaultProperty()
  33. {
  34. returnnull;
  35. }
  36. objectICustomTypeDescriptor.GetEditor(TypeeditorBaseType)
  37. {
  38. returnTypeDescriptor.GetEditor(this,editorBaseType,true);
  39. }
  40. EventDescriptorCollectionICustomTypeDescriptor.GetEvents()
  41. {
  42. returnTypeDescriptor.GetEvents(this,true);
  43. }
  44. EventDescriptorCollectionICustomTypeDescriptor.GetEvents(Attribute[]attributes)
  45. {
  46. returnTypeDescriptor.GetEvents(this,attributes,true);
  47. }
  48. PropertyDescriptorCollectionICustomTypeDescriptor.GetProperties()
  49. {
  50. return((ICustomTypeDescriptor)this).GetProperties(newAttribute[0]);
  51. }
  52. PropertyDescriptorCollectionICustomTypeDescriptor.GetProperties(Attribute[]attributes)
  53. {
  54. ArrayListprops=newArrayList();
  55. TypethisType=this.GetType();
  56. PropertyInfo[]pis=thisType.GetProperties();
  57. foreach(PropertyInfopinpis)
  58. {
  59. if(p.DeclaringType==thisType||p.PropertyType.ToString()=="System.Drawing.Color")
  60. {
  61. //判断属性是否显示
  62. BrowsableAttributeBrowsable=(BrowsableAttribute)Attribute.GetCustomAttribute(p,typeof(BrowsableAttribute));
  63. if(Browsable!=null)
  64. {
  65. if(Browsable.Browsable==true||p.PropertyType.ToString()=="System.Drawing.Color")
  66. {
  67. PropertyStubpsd=newPropertyStub(p,attributes);
  68. props.Add(psd);
  69. }
  70. }
  71. else
  72. {
  73. PropertyStubpsd=newPropertyStub(p,attributes);
  74. props.Add(psd);
  75. }
  76. }
  77. }
  78. PropertyDescriptor[]propArray=(PropertyDescriptor[])props.ToArray(typeof(PropertyDescriptor));
  79. returnnewPropertyDescriptorCollection(propArray);
  80. }
  81. objectICustomTypeDescriptor.GetPropertyOwner(PropertyDescriptorpd)
  82. {
  83. returnthis;
  84. }
  85. #endregion
  86. }
  87. ///<summary>
  88. ///下面这段代码来源于:http://www.bluespace.cn/Html/Csdn/2_47/View_4702219.html
  89. ///</summary>
  90. #regionPropertyStub定义
  91. publicclassPropertyStub:PropertyDescriptor
  92. {
  93. PropertyInfoinfo;
  94. publicPropertyStub(PropertyInfopropertyInfo,Attribute[]attrs)
  95. :base(propertyInfo.Name,attrs)
  96. {
  97. this.info=propertyInfo;
  98. }
  99. publicoverrideTypeComponentType
  100. {
  101. get{returnthis.info.ReflectedType;}
  102. }
  103. publicoverrideboolIsReadOnly
  104. {
  105. get{returnthis.info.CanWrite==false;}
  106. }
  107. publicoverrideTypePropertyType
  108. {
  109. get{returnthis.info.PropertyType;}
  110. }
  111. publicoverrideboolCanResetValue(objectcomponent)
  112. {
  113. returnfalse;
  114. }
  115. publicoverrideobjectGetValue(objectcomponent)
  116. {
  117. //Console.WriteLine("GetValue:"+component.GetHashCode());
  118. try
  119. {
  120. returnthis.info.GetValue(component,null);
  121. }
  122. catch
  123. {
  124. returnnull;
  125. }
  126. }
  127. publicoverridevoidResetValue(objectcomponent)
  128. {
  129. }
  130. publicoverridevoidSetValue(objectcomponent,objectvalue)
  131. {
  132. //Console.WriteLine("SetValue:"+component.GetHashCode());
  133. this.info.SetValue(component,value,null);
  134. }
  135. publicoverrideboolShouldSerializeValue(objectcomponent)
  136. {
  137. returnfalse;
  138. }
  139. //通过重载下面这个属性,可以将属性在PropertyGrid中的显示设置成中文
  140. publicoverridestringDisplayName
  141. {
  142. get
  143. {
  144. if(info!=null)
  145. {
  146. MyControlAttibuteuicontrolattibute=(MyControlAttibute)Attribute.GetCustomAttribute(info,typeof(MyControlAttibute));
  147. if(uicontrolattibute!=null)
  148. returnuicontrolattibute.PropertyName;
  149. else
  150. {
  151. returninfo.Name;
  152. }
  153. }
  154. else
  155. return"";
  156. }
  157. }
  158. }
  159. #endregion

修改后的Form界面如下:

C#基础系列:开发自己的窗体设计器(PropertyGrid显示中文属性名)Form中由于我仅仅是演示了TextBox所以我就是直接将创建TextBoxProperty对象的代码放在了Control_Click中:

//我这里仅仅做TextBox的属性演示,如果是其它的控件的话,那么你需要设计不同的ControlProperty(比如TextBoxPropertyComboBoxProperty

if (sender is TextBox)

{

this.propertyGrid1.SelectedObject = new TextBoxProperty((TextBox)sender);

}

else

{

this.propertyGrid1.SelectedObject = null;

}

对于实际的需要来说应该是根据Control的类型,通过工厂模式来创建ControlProperty

Form的整个代码如下,窗体如上图所示:

  1. //在Form中增加几个Button,分别命名为cmdArrow,cmdLabel,cmdTextBox,cmdComboBox,cmdGroupBox
  2. publicpartialclassForm1:Form
  3. {
  4. privateMouseHook_MouseHook;
  5. //我们将所有的已经与具体控件关联了的UISizeKnob缓存在这个HashTable中
  6. privateHashtable_HashUISizeKnob;
  7. //负责控件移动的类
  8. privateHashtable_HashUIMoveKnob;
  9. publicForm1()
  10. {
  11. InitializeComponent();
  12. this._MouseHook=newMouseHook(this);
  13. this._HashUISizeKnob=newHashtable();
  14. this._HashUIMoveKnob=newHashtable();
  15. //为了简洁明了,我们在ControlAdded中来设置具体控件和UISizeKnob的关联
  16. this.ControlAdded+=newControlEventHandler(Form1_ControlAdded);
  17. }
  18. voidForm1_ControlAdded(objectsender,ControlEventArgse)
  19. {
  20. if(!(e.ControlisUISizeDot))
  21. {
  22. this._HashUISizeKnob.Add(e.Control,newUISizeKnob(e.Control));
  23. this._HashUIMoveKnob.Add(e.Control,newUIMoveKnob(e.Control));
  24. //点击控件的时候,显示控件的选择
  25. e.Control.Click+=newEventHandler(Control_Click);
  26. }
  27. }
  28. voidControl_Click(objectsender,EventArgse)
  29. {
  30. //寿险清除已经选择的控件
  31. foreach(UISizeKnobknobinthis._HashUISizeKnob.Values)
  32. {
  33. knob.ShowUISizeDots(false);
  34. }
  35. //System.ComponentModel.Design.ISelectionService
  36. //System.Drawing.Design.IToolboxService
  37. try
  38. {
  39. ((UISizeKnob)this._HashUISizeKnob[sender]).ShowUISizeDots(true);
  40. //我这里仅仅做TextBox的属性演示,如果是其它的控件的话,那么你需要设计不同的ControlProperty(比如TextBoxProperty,ComboBoxProperty)
  41. if(senderisTextBox)
  42. {
  43. this.propertyGrid1.SelectedObject=newTextBoxProperty((TextBox)sender);
  44. }
  45. else
  46. {
  47. this.propertyGrid1.SelectedObject=null;
  48. }
  49. }
  50. catch{}
  51. }
  52. privatevoidcmdArrow_Click(objectsender,EventArgse)
  53. {
  54. SettingService.Instance.SelectedToolBoxControl=null;
  55. }
  56. privatevoidcmdLabel_Click(objectsender,EventArgse)
  57. {
  58. SettingService.Instance.SelectedToolBoxControl=newLabel();
  59. }
  60. privatevoidcmdTextBox_Click(objectsender,EventArgse)
  61. {
  62. SettingService.Instance.SelectedToolBoxControl=newTextBox();
  63. }
  64. privatevoidcmdComboBox_Click(objectsender,EventArgse)
  65. {
  66. SettingService.Instance.SelectedToolBoxControl=newComboBox();
  67. }
  68. privatevoidcmdGroupBox_Click(objectsender,EventArgse)
  69. {
  70. SettingService.Instance.SelectedToolBoxControl=newGroupBox();
  71. }
  72. }

好了,让

PropertyGrid 显示中文属性的思路和代码都给了,希望能够给你点启发和帮助。

相关文章:

C#基础系列:开发自己的窗体设计器(总纲)

C#基础系列:开发自己的窗体设计器(在容器上拖动鼠标增加控件)

C#基础系列:开发自己的窗体设计器(实现控件的选择)

C#基础系列:开发自己的窗体设计器(实现控件的拖动)

你可能感兴趣的:(设计模式,C++,c,C#,Office)