ASP.NET控件开发基础之自定义控件浅析

开发 后端
ASP.NET控件开发基础之自定义控件向你讲述了我们在ASP.NET控件开发基础中自定义控件方面的情况。

ASP.NET本身提供了很多控件,提供给我们这些比较懒惰的人使用,我认为控件的作用就在此,因为我们不想重复工作,所以要创建它,这个本身便是一个需求的关系,所以学习自定义控件开发很有意思.

wrox网站上有本书 Professional ASP.NET 2.0 Server Control and Component Development

现在还没有出版,但网站上放出了代码,所以正好下载过来学习一下.

我看过前几章代码,环环相扣,作者用不同的知识向我们展示同一个效果,所以循序渐进的学下来很有好处.

虽然自己对控件开发还不是很熟悉,但我感觉对于ASP.NET控件开发基础中自定义控件以下几点很重要,是我自己总结的

1.了解ASP.NET自定义控件之间的继承关系

***是先看看看System.Web.UI命名空间

(1)Control 类,所有的控件都共享的一个类,你需要去看下其里面受保护的几个方法和属性,虽然一下看不完,以后会发现常常用到这些方法

大家可以在MSDN看一下其派生类

(2)HtmlTextWriter 类

不得不了解的一个类,主要工作就是我们写的标记字符和文本输出

2.ASP.NET自定义控件之重写方法

(1) 必须继承Control类

(2) 重写Control类的Render方法,这个是必须的,因为其他控件都继承了Control 类类,所以几乎所有控件都有这个方法

3.熟悉元数据

大家都知道ASP.NET控件属性在编辑器上是分类的,如外观,行为,布局等,每个属性还给出了解释

简单的元数据就是起到这个作用,当然你也可以不加,但使用了元数据让人感到有亲切感,写法如

[CategoryAttribute("Appearance")]

要使用元数据,必须引用System.ComponentModel命名控件,一般你如果写组件的话,不可能不用到这样类库。具体的MSDN上有所介绍。

一.输出字符串

说多了没意思,还是来演练吧。首先你得了解HTML。来看下面代码,效果就是输出HTML到客户端

示例一

  1. using System;  
  2. using System.Web.UI;  
  3.  
  4. namespace CustomComponents  
  5. {  
  6.   /**//// ﹤summary﹥  
  7.   /// Summary description for CreditCardForm  
  8.   /// ﹤/summary﹥  
  9.   public class CreditCardForm1 : Control  
  10.   {  
  11.     protected override void Render(HtmlTextWriter writer)  
  12.     {  
  13.       writer.Write("﹤table style='width:287px;height:124px;border-width:0;'﹥");  
  14.       writer.Write("﹤tr﹥");  
  15.       writer.Write("﹤td﹥﹤strong﹥Payment Method﹤/strong﹥﹤/td﹥");  
  16.       writer.Write("﹤td﹥");  
  17.       writer.Write("﹤select name='PaymentMethod' id='PaymentMethod' style='width:100%;'﹥");  
  18.       writer.Write("﹤option value='0'﹥Visa﹤/option﹥");  
  19.       writer.Write("﹤option value='1'﹥MasterCard﹤/option﹥");  
  20.       writer.Write("﹤/select﹥");  
  21.       writer.Write("﹤/td﹥");  
  22.       writer.Write("﹤/tr﹥");  
  23.       writer.Write("﹤tr﹥");  
  24.       writer.Write("﹤td﹥﹤strong﹥Credit Card No.﹤/strong﹥﹤/td﹥");  
  25.       writer.Write("﹤td﹥﹤input name='CreditCardNo' id='CreditCardNo' type='text' /﹥﹤/td﹥");  
  26.       writer.Write("﹤/tr﹥");  
  27.       writer.Write("﹤tr﹥");  
  28.       writer.Write("﹤td﹥﹤strong﹥Cardholder's Name﹤/strong﹥﹤/td﹥");  
  29.       writer.Write("﹤td﹥﹤input name='CardholderName' id='CardholderName' type='text' /﹥﹤/td﹥");  
  30.       writer.Write("﹤/tr﹥");  
  31.       writer.Write("﹤tr﹥");  
  32.       writer.Write("﹤td﹥﹤strong﹥Expiration Date﹤/strong﹥﹤/td﹥");  
  33.       writer.Write("﹤td﹥");  
  34.       writer.Write("﹤select name='Month' id='Month'﹥");  
  35.       for (int day = 1; day ﹤ 13; day++)  
  36.       {  
  37.         if (day ﹤ 10)  
  38.           writer.Write("﹤option value='" + day.ToString() + "'﹥" + "0" + day.ToString() + "﹤/option﹥");  
  39.         else 
  40.           writer.Write("﹤option value='" + day.ToString() + "'﹥" + day.ToString() + "﹤/option﹥");  
  41.       }  
  42.       writer.Write("﹤/select﹥");  
  43.       writer.Write(" ");  
  44.       writer.Write("﹤select name='Year' id='Year'﹥");  
  45.       for (int year = 2005; year ﹤ 2015; year++)  
  46.       {  
  47.         writer.Write("﹤option value='" + year.ToString() + "'﹥" + year.ToString() + "﹤/option﹥");  
  48.       }  
  49.       writer.Write("﹤/select﹥");  
  50.       writer.Write("﹤/td﹥");  
  51.       writer.Write("﹤/tr﹥");  
  52.       writer.Write("﹤tr﹥");  
  53.       writer.Write("﹤td align='center' colspan='2'﹥");  
  54.       writer.Write("﹤input type='submit' value='Submit' /﹥");  
  55.       writer.Write("﹤/td﹥");  
  56.       writer.Write("﹤/tr﹥");  
  57.       writer.Write("﹤/table﹥");  
  58.  
  59.       base.Render(writer);  
  60.     }  
  61.   }  

效果很简单,其实就一直在输出HTML再加几个属性,大家可以直接把代码放在App_Code文件夹里,就可自动编译,当然也可以创建web控件库.
注意要继承Control类,重写Render方法,用HtmlTextWriter类的Write输出HTML

使用控件

(1).需要先注册一下

﹤%@ Register TagPrefix="custom" Namespace="CustomComponents" %﹥

(2) 然后就使用标签输出效果

﹤custom:CreditCardForm1 runat="server" ID="ccf" /﹥

下为效果图

ASP.NET自定义控件之控件效果图 

二.改善,加入属性和元数据

可能上面做出的 控件毫无用处,但却可以让你熟悉一下步骤,上面的控件定的很死,没有定义任何属性,用处不大,下面来改造

我们来定义常用属性,然后再输出,这样我们就可以修改控件的属性了,

示例二

  1. using System;  
  2. using System.Web.UI;  
  3. using System.ComponentModel;  
  4.  
  5. namespace CustomComponents  
  6. {  
  7.   [DefaultPropertyAttribute("CardholderNameText")]  
  8.     [ToolboxData(@"﹤{0}:CreditCardForm2   
  9.     PaymentMethodText='信用卡类型' CreditCardNoText='信用卡卡号'   
  10.     CardholderNameText='信用卡持有者姓名' SubmitButtonText = '提交'    
  11.     runat='server'﹥﹤/{0}:CreditCardForm2﹥")  
  12.     ]  
  13.   public class CreditCardForm2 : Control  
  14.   {  
  15.     private string paymentMethodText = "信用卡类型";  
  16.     private string creditCardNoText = "信用卡卡号";  
  17.     private string cardholderNameText = "信用卡持有者姓名";  
  18.     private string expirationDateText = "***使用时间";  
  19.     private string submitButtonText = "提交";  
  20.  
  21.     [BrowsableAttribute(true)]  
  22.     [DescriptionAttribute("获取和设置信用卡类型")]  
  23.       [DefaultValueAttribute("信用卡类型")]  
  24.     [CategoryAttribute("Appearance")]  
  25.     public virtual string PaymentMethodText  
  26.     {  
  27.       get { return this.paymentMethodText; }  
  28.       set { this.paymentMethodText = value; }  
  29.     }  
  30.  
  31.     [BrowsableAttribute(true)]  
  32.     [DescriptionAttribute("获取或设置信用卡卡号")]  
  33.     [DefaultValueAttribute("信用卡卡号")]  
  34.     [CategoryAttribute("Appearance")]  
  35.     public virtual string CreditCardNoText  
  36.     {  
  37.       get { return this.creditCardNoText; }  
  38.       set { this.creditCardNoText = value; }  
  39.     }  
  40.  
  41.     [BrowsableAttribute(true)]  
  42.       [DescriptionAttribute("获取或设置信用卡持有者姓名")]  
  43.     [DefaultValueAttribute("信用卡持有者姓名")]  
  44.     [CategoryAttribute("Appearance")]  
  45.     public virtual string CardholderNameText  
  46.     {  
  47.       get { return this.cardholderNameText; }  
  48.       set { this.cardholderNameText = value; }  
  49.     }  
  50.  
  51.     [BrowsableAttribute(true)]  
  52.       [DescriptionAttribute("获取或设置***使用时间")]  
  53.       [DefaultValueAttribute("***使用时间")]  
  54.     [CategoryAttribute("Appearance")]  
  55.     public virtual string ExpirationDateText  
  56.     {  
  57.       get { return this.expirationDateText; }  
  58.       set { this.expirationDateText = value; }  
  59.     }  
  60.  
  61.     [BrowsableAttribute(true)]  
  62.     [DescriptionAttribute("获取或设置按钮标签")]  
  63.     [DefaultValueAttribute("提交")]  
  64.     [CategoryAttribute("Appearance")]  
  65.     public virtual string SubmitButtonText  
  66.     {  
  67.       get { return this.submitButtonText; }  
  68.       set { this.submitButtonText = value; }  
  69.     }  
  70.  
  71.     protected override void Render(HtmlTextWriter writer)  
  72.     {  
  73.       writer.Write("﹤table style='width:287px;height:124px;border-width:0;'﹥");  
  74.       writer.Write("﹤tr﹥");  
  75.       writer.Write("﹤td﹥" + PaymentMethodText + "﹤/td﹥");  
  76.       writer.Write("﹤td﹥");  
  77.       writer.Write("﹤select name='PaymentMethod' id='PaymentMethod' style='width:100%;'﹥");  
  78.       writer.Write("﹤option value='0'﹥Visa﹤/option﹥");  
  79.       writer.Write("﹤option value='1'﹥MasterCard﹤/option﹥");  
  80.       writer.Write("﹤/select﹥");  
  81.       writer.Write("﹤/td﹥");  
  82.       writer.Write("﹤/tr﹥");  
  83.       writer.Write("﹤tr﹥");  
  84.       writer.Write("﹤td﹥" + CreditCardNoText + "﹤/td﹥");  
  85.       writer.Write("﹤td﹥﹤input name='CreditCardNo' id='CreditCardNo' type='text' /﹥﹤/td﹥");  
  86.       writer.Write("﹤/tr﹥");  
  87.       writer.Write("﹤tr﹥");  
  88.       writer.Write("﹤td﹥" + CardholderNameText + "﹤/td﹥");  
  89.       writer.Write("﹤td﹥﹤input name='CardholderName' id='CardholderName' type='text' /﹥﹤/td﹥");  
  90.       writer.Write("﹤/tr﹥");  
  91.       writer.Write("﹤tr﹥");  
  92.       writer.Write("﹤td﹥" + ExpirationDateText + "﹤/td﹥");  
  93.       writer.Write("﹤td﹥");  
  94.       writer.Write("﹤select name='Month' id='Month'﹥");  
  95.       for (int day = 1; day ﹤ 13; day++)  
  96.       {  
  97.         if (day ﹤ 10)  
  98.           writer.Write("﹤option value='" + day.ToString() + "'﹥" + "0" + day.ToString() + "﹤/option﹥");  
  99.         else 
  100.           writer.Write("﹤option value='" + day.ToString() + "'﹥" + day.ToString() + "﹤/option﹥");  
  101.       }  
  102.       writer.Write("﹤/select﹥");  
  103.       writer.Write(" ");  
  104.       writer.Write("﹤select name='Year' id='Year'﹥");  
  105.       for (int year = 2005; year ﹤ 2015; year++)  
  106.       {  
  107.         writer.Write("﹤option value='" + year.ToString() + "'﹥" + year.ToString() + "﹤/option﹥");  
  108.       }  
  109.       writer.Write("﹤/select﹥");  
  110.       writer.Write("﹤/td﹥");  
  111.       writer.Write("﹤/tr﹥");  
  112.       writer.Write("﹤tr﹥");  
  113.       writer.Write("﹤td align='center' colspan='2'﹥");  
  114.       writer.Write("﹤input type='submit' value='" + SubmitButtonText + "' /﹥");  
  115.       writer.Write("﹤/td﹥");  
  116.       writer.Write("﹤/tr﹥");  
  117.       writer.Write("﹤/table﹥");  
  118.  
  119.       base.Render(writer);  
  120.     }  
  121.   }  

上面我们接触到了元数据了,意思应该很好理解,为了测试元数据的作用,大家可以新建一个类库项目,然后把写的代码放这个项目里面,接着web网站引用这个项目,成功生成以后,你会发现工具箱已经自动帮你加上了这几个控件

自动添加的控件 

接着你要做的工作就是拖动你需要的控件,然后你会在属性面板看到下图

拖动所需要的控件 

然后你再结合代码中的元数据,应该就知道大概意思了.(可以根据你的理解结合MSDN看)

三.再次改善,淘汰用Write方法以字符串的方式输出HTML

接着我们继续发现问题,我们发现我们除了定义几个需要自己来修改的属性外,还是要用来大量的字符串用来输出HTML,而且容易输错.所以HtmlTextWriter类提供几个有用的方法用来代替.

(1)AddStyleAttribute方法 为标签添加样式属性

(2)AddAttribute方法        为标签添加属性

(3)RenderBeginTag          开始写入标签头 如﹤table....﹥

(4)RenderEndTag            写入标签尾部,如﹤/table﹥

这里有几点需要特别注意.

一.因为其定义方式跟我们平时定义方式不同,我们平时写HTML时,是先写标签开头,再写标签的属性.如﹤table borderwidth="0"﹥,然而我们在使用上面几个方法时,需要有先后顺序,我们需要先定义标签的属性和样式,然后再输出标签头.

二.标签头和尾,需一一对应.可以理解为嵌套关系.***的理解方法就是输出代码后,查看源文件,再结合原来定义的代码来看.

还是看代码比较容易说明,由于CreditCardForm2已经定义了我们需要的属性,而我们现在要做的只是用标签的形式来替代字符串的形式,所以只需要继承CreditCardForm2类,重写Render方法即可

示例三

  1. protected override void Render(HtmlTextWriter writer)  
  2.     {  
  3.       writer.AddStyleAttribute(HtmlTextWriterStyle.BorderWidth, "0");  
  4.       writer.RenderBeginTag(HtmlTextWriterTag.Table);  
  5.       writer.RenderBeginTag(HtmlTextWriterTag.Tr);  
  6.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  7.       writer.Write("﹤strong﹥" + PaymentMethodText + "﹤/strong﹥");  
  8.       writer.RenderEndTag();  
  9.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  10.       writer.AddAttribute(HtmlTextWriterAttribute.Name, "PaymentMethod");  
  11.       writer.AddAttribute(HtmlTextWriterAttribute.Id, "PaymentMethod");  
  12.       writer.AddStyleAttribute(HtmlTextWriterStyle.Width, "100%");  
  13.       writer.RenderBeginTag(HtmlTextWriterTag.Select);  
  14.  
  15.       writer.AddAttribute(HtmlTextWriterAttribute.Value, "0");  
  16.       writer.RenderBeginTag(HtmlTextWriterTag.Option);  
  17.       writer.Write("Visa");  
  18.       writer.RenderEndTag();  
  19.  
  20.       writer.AddAttribute(HtmlTextWriterAttribute.Value, "1");  
  21.       writer.RenderBeginTag(HtmlTextWriterTag.Option);  
  22.       writer.Write("MasterCard");  
  23.       writer.RenderEndTag();  
  24.  
  25.       writer.RenderEndTag();  
  26.       writer.RenderEndTag();  
  27.       writer.RenderEndTag();  
  28.  
  29.       writer.RenderBeginTag(HtmlTextWriterTag.Tr);  
  30.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  31.       writer.Write("﹤strong﹥" + CreditCardNoText + "﹤/strong﹥");  
  32.       writer.RenderEndTag();  
  33.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  34.       writer.AddAttribute(HtmlTextWriterAttribute.Name, "CreditCardNo");  
  35.       writer.AddAttribute(HtmlTextWriterAttribute.Id, "CreditCardNo");  
  36.       writer.AddAttribute(HtmlTextWriterAttribute.Type, "text");  
  37.       writer.RenderBeginTag(HtmlTextWriterTag.Input);  
  38.       writer.RenderEndTag();  
  39.       writer.RenderEndTag();  
  40.       writer.RenderEndTag();  
  41.  
  42.       writer.RenderBeginTag(HtmlTextWriterTag.Tr);  
  43.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  44.       writer.Write("﹤strong﹥" + CardholderNameText + "﹤/strong﹥");  
  45.       writer.RenderEndTag();  
  46.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  47.       writer.AddAttribute(HtmlTextWriterAttribute.Name, "CardholderName");  
  48.       writer.AddAttribute(HtmlTextWriterAttribute.Id, "CardholderName");  
  49.       writer.AddAttribute(HtmlTextWriterAttribute.Type, "text");  
  50.       writer.RenderBeginTag(HtmlTextWriterTag.Input);  
  51.       writer.RenderEndTag();  
  52.       writer.RenderEndTag();  
  53.       writer.RenderEndTag();  
  54.  
  55.       writer.RenderBeginTag(HtmlTextWriterTag.Tr);  
  56.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  57.       writer.Write("﹤strong﹥" + ExpirationDateText + "﹤/strong﹥");  
  58.       writer.RenderEndTag();  
  59.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  60.       writer.AddAttribute(HtmlTextWriterAttribute.Name, "Month");  
  61.       writer.AddAttribute(HtmlTextWriterAttribute.Id, "Month");  
  62.       writer.RenderBeginTag(HtmlTextWriterTag.Select);  
  63.  
  64.       for (int day = 1; day ﹤ 13; day++)  
  65.       {  
  66.         writer.AddAttribute(HtmlTextWriterAttribute.Value, day.ToString());  
  67.         writer.RenderBeginTag(HtmlTextWriterTag.Option);  
  68.  
  69.         if (day ﹤ 10)  
  70.           writer.Write("0" + day.ToString());  
  71.         else 
  72.           writer.Write(day);  
  73.  
  74.         writer.RenderEndTag();  
  75.       }  
  76.  
  77.       writer.RenderEndTag();  
  78.       writer.Write(" ");  
  79.  
  80.       writer.AddAttribute(HtmlTextWriterAttribute.Name, "Year");  
  81.       writer.AddAttribute(HtmlTextWriterAttribute.Id, "Year");  
  82.       writer.RenderBeginTag(HtmlTextWriterTag.Select);  
  83.  
  84.       for (int year = 2005; year ﹤ 2015; year++)  
  85.       {  
  86.         writer.AddAttribute(HtmlTextWriterAttribute.Value, year.ToString());  
  87.         writer.RenderBeginTag(HtmlTextWriterTag.Option);  
  88.         writer.Write(year);  
  89.         writer.RenderEndTag();  
  90.       }  
  91.  
  92.       writer.RenderEndTag();  
  93.  
  94.       writer.RenderEndTag();  
  95.       writer.RenderEndTag();  
  96.  
  97.       writer.RenderBeginTag(HtmlTextWriterTag.Tr);  
  98.       writer.AddAttribute(HtmlTextWriterAttribute.Align, "center");  
  99.       writer.AddAttribute(HtmlTextWriterAttribute.Colspan, "2");  
  100.       writer.RenderBeginTag(HtmlTextWriterTag.Td);  
  101.       writer.AddAttribute(HtmlTextWriterAttribute.Type, "submit");  
  102.       writer.AddAttribute(HtmlTextWriterAttribute.Value, SubmitButtonText);  
  103.       writer.RenderBeginTag(HtmlTextWriterTag.Input);  
  104.       writer.RenderEndTag();  
  105.       writer.RenderEndTag();  
  106.       writer.RenderEndTag();  
  107.       writer.RenderEndTag();  
  108.     } 

实现的效果虽然一样,但上面的代码是不是漂亮很多,而且不容易输错.这也是所提倡的做法

四.未使用视图状态的后果

还是视图状态,关于视图状态大家可以参考MSDN和相关文章

看以下的示例,还是CreditCardForm3这个控件

  1. if (!IsPostBack)  
  2.     {  
  3.       creditcardform.CardholderNameText = "Full Name";  
  4.       creditcardform.CreditCardNoText = "CreditCardNo";  
  5.       creditcardform.ExpirationDateText = "ExpirationDate";  
  6.       creditcardform.PaymentMethodText = "Payment Options";  
  7.       creditcardform.SubmitButtonText = "Send";  
  8.     } 

***加载效果

控件加载效果 

点击按钮以后

点击按钮后的效果 

五.使用视图状态改善效果

前提条件是你未禁用视图状态

继承CreditCardForm3,改写每个属性

  1. public override string PaymentMethodText  
  2.     {  
  3.         get { return ViewState["PaymentMethodText"] != null ? (string)ViewState["PaymentMethodText"] : "信用卡类型"; }  
  4.       set { ViewState["PaymentMethodText"] = value; }  
  5.     }  
  6.  
  7.     public override string CreditCardNoText  
  8.     {  
  9.         get { return ViewState["CreditCardNoText"] != null ? (string)ViewState["CreditCardNoText"] : "信用卡卡号"; }  
  10.       set { ViewState["CreditCardNoText"] = value; }  
  11.     }  
  12.  
  13.     public override string CardholderNameText  
  14.     {  
  15.         get { return ViewState["CardholderNameText"] != null ? (string)ViewState["CardholderNameText"] : "信用卡持有者姓名"; }  
  16.       set { ViewState["CardholderNameText"] = value; }  
  17.     }  
  18.  
  19.     public override string ExpirationDateText  
  20.     {  
  21.         get { return ViewState["ExpirationDateText"] != null ? (string)ViewState["ExpirationDateText"] : "***使用时间"; }  
  22.       set { ViewState["ExpirationDateText"] = value; }  
  23.     }  
  24.  
  25.     public override string SubmitButtonText  
  26.     {  
  27.         get { return ViewState["SubmitButtonText"] != null ? (string)ViewState["SubmitButtonText"] : "提交"; }  
  28.       set { ViewState["SubmitButtonText"] = value; }  
  29.     } 

以上全为个人见解,如有错误,希望大家指出.

ASP.NET控件开发基础之自定义控件的了解就向大家介绍到这里,对于ASP.NET控件开发基础在以后的文章中还会想大家慢慢介绍。

责任编辑:仲衡 来源: 博客园
相关推荐

2009-08-06 17:52:45

ASP.NET控件开发自定义控件

2009-07-28 09:32:41

ASP.NET自定义控

2009-08-06 17:13:56

ASP.NET自定义控

2009-08-10 14:16:59

ASP.NET自定义控

2009-08-07 11:12:58

ASP.NET控件开发

2009-07-31 10:23:09

ASP.NET源码DateTimePic

2009-08-06 18:18:27

ASP.NET控件开发ASP.NET复合控件

2011-04-19 10:33:16

ASP.NET自定义控

2009-08-07 14:05:21

ASP.NET控件

2009-08-06 15:21:45

ASP.NET控件开发RenderConte

2009-08-07 15:24:16

ASP.NET模板控件

2009-07-28 16:21:03

Asp.net AjaAutoComplet

2009-08-06 13:08:23

ASP.NET控件开发

2009-08-06 18:32:00

ASP.NET控件开发ASP.NET复合控件

2009-08-05 18:46:21

ComboBox显示ASP.NET控件开发

2009-08-05 17:11:51

ASP.NET控件开发ASP.NET服务器控

2009-07-27 17:25:53

ASP.NET验证控件

2009-08-07 15:34:15

ASP.NET数据绑定

2009-08-05 18:32:28

HtmlTextWriASP.NET控件开发

2009-08-07 17:41:07

ASP.NET Web
点赞
收藏

51CTO技术栈公众号