关于自定义WEB服务器控件的知识与实例  

 

本文重点介绍一种实现控件呈现的常用方法--使用WebControl类的RenderContents方法实现控件呈现。

   基础知识

    就服务器控件而言只存在两种情况:一种是具有外观可视化元素的控件,还有一种是不具有外观可视化元素的控件。如果需要开发的服务器控件包含可视化元素, 那么多数情况下,建议开发人员创建继承自System.Web.UI.WebControls.WebControl基类的控件类。这种做法的主要原因是 基于便捷性考虑。因为,WebControl类可提供服务器控件的部分与外观有关的公共属性、方法和事件等。通过该类定义的属性,可以控制服务器控件的外 观和行为。例如,使用BackColor和ForeColor属性,可以分别控制服务器控件的背景色和前景颜色;在可以显示边框的控件上,可以通过设置 BorderWidth、BorderStyle和BorderColor属性,控制边框宽度、边框样式和边框颜色;服务器控件的大小可以通过 Height和Width属性来指定等等。如果控件基类是Control类,那么实现这些类似内容则非常繁琐。

   在使用WebControl基类实现控件呈现的过程中,必然要使用该类所提供的属性和方法等成员对象。这是读者需要重点掌握的内容。另外,对于该基类的构造函数也是不容忽视的。下面首先从WebControl的构造函数开始入手进行讲解,随后将说明常见的成员对象。

   WebControl类包括三个构造函数,它们都用于初始化WebControl类的新实例,然而它们之间还存在一些细小的差异。

   (1)protected WebControl ()

   该构造函数用于初始化表示Span HTML元素的WebControl类的新实例。通常情况下,开发人员并不直接调用此构造函数。相反,它通常由派生类的构造函数调用以将TagKey属性初始化为Span枚举值。在随后的示例中,将重写TagKey属性,从而调用此构造函数。

 (2)public WebControl (HtmlTextWriterTag tag)

   开发人员可使用此构造函数创建并初始化使用指定的System.Web.UI.HtmlTextWriterTag值的WebControl类的新实例。其中的参数tag表示HtmlTextWriterTag枚举值之一。可能读者对于HtmlTextWriterTag还不太熟悉。它是一个枚举类型,其枚举值多为HTML标记,例如,A、B、Bold、Button等等。

   (3)protected WebControl (string tag)

   使用此构造函数可创建并初始化使用指定的HTML标记的WebControl类的新实例。其中参数tag表示HTML标记。当使用该构造函数时一定要注意:不能直接调用此构造函数。相反,它通常由派生类的构造函数调用以初始化TagKey和TagName属性

   在了解了WebControl类的构造函数之后,读者还必须了解WebControl类的一些常用属性和方法。下面列举了这些常用成员对象,它们对于实现控件呈现有着重要意义。

   (1)Attributes属性

   该属性用于获取与控件的属性不对应的任意特性(只用于呈现)的集合,其属性类型为AttributeCollection。

   (2)ControlStyle属性

   该属性用于获取服务器控件的样式,它是Style类型。ControlStyle属性封装WebControl类的所有外观属性,如BorderColor和Font。

   (3)TagKey属性

    该属性用于获取与此服务器控件相对应的System.Web.UI.HtmlTextWriterTag值,其属性类型为HtmlTextWriterTag枚举。

   (4)protected virtual void AddAttributeToRender(HtmlTextWriter writer);

    该方法将需要呈现的HTML属性和样式添加到指定的System.Web.UI.HtmlTextWriter中。注意在重写过程中,一定要调用基类中相应的方法

   (5)public void ApplyStyle(Style s);

         该方法将指定样式的所有非空白元素复制到控件,改写控件的所有现有的样式元素。

   (6)public void MergeStyle(Style s);

    该方法将指定样式的所有非空白元素复制到控件,但不改写该控件现有的任何样式元素。

   (7)protected override void Render(HtmlTextWriter writer);方法

    该方法重写了Control.Render。

   (8)protected virtual void RenderContents(HtmlTextWriter writer);

   该方法将控件的内容呈现到指定的编写器中。如果要在控件的标签中写入文本或其他内容,则需要重写该方法;如果要使用默认逻辑来呈现子控件,那么一定要调用基类中相应的方法。

    可能读者已经注意到WebControl基类中包括的两个方法:Render和RenderContents。根据上文所介绍的内容可 知,Control基类中包括Render方法。由于WebControl类继承自Control类,因此,WebControl类中包含Render方 法是无可非议的。然而,WebControl类中却有一个RenderContents方法,并且该方法与Render方法在功能、参数等方面都非常相 似。那么在呈现控件过程中到底应该使用哪一个呢?

   实际上,在通常情况下,如果服务器控件自WebControl基类派生,那么其中的Render方法很少使用,而主要使用RenderContents方法实现控件呈现。为了说明其中的原因,我们必须了解WebControl基类中Render方法的实现逻辑。

   在WebControl基类中的Render方法的实现示意性代码如下所示:

 protected override void Render(HtmlTextWriter output)
 {
  RenderBeginTag(output);
  RenderContents(output);
  RenderEndTag(output);
 }

   在WebControl基类中的RenderBeginTag方法的实现示意性代码如下:

 public virtual void RenderBeginTag(HtmlTextWriter output)
 {
  AddAttributesToRender(output);              //将要呈现的控件的属性添加到HtmlTextWriter对象(流)中
  HtmlTextWriterTag tagKey = TagKey;
  if(tagKey != HtmlTextWriterTag.Unknown)
  {
 output.RenderBeginTag(tagKey);
  } else {
 output.RenderBeginTag(this.TagName);
  }
 }


   在WebControl基类中的RenderContents方法的实现示意性代码如下:

 protected override void RenderContents(HtmlTextWriter output){
  //使用默认逻辑来呈现子控件,那么一定要调用基类中的方法。
  base.Render(output);
 }
   分析以上代码可以得出以下结论:

   一、为了在由WebControl派生的类中实现控件呈现,必须重写AddAttributesToRender、RenderBeginTag、RenderEndTag、RenderContents等方法中的一个或者多个,而不必重写Render方法

    二、重写AddAttributesToRender、RenderBeginTag、RenderEndTag、RenderContents等方法 非常重要(请注意重写这些方法的条件及注意事项),否则服务器控件可能会出现丢失标签的情况,这将严重影响服务器控件的呈现。

   三、当呈现服务器控件标签中的内容时,必须重写RenderContents方法

   上文介绍了WebControl类的一些基本知识。尤其是对于上文所列举的示意性代码需要重点理解。这对于实现控件呈现有着重要作用。

   应用示例

    相信读者在浏览各个网站时,经常会看到"联系我们"等类似文字。当单击这些文字时,操作系统将自动打开自身所带的邮件客户端软件,提示用户发送邮件。本 例将实现一个自定义服务器控件RenderContentsControl,用于通过呈现包含"mailto:邮件地址"的超链接。超链接文字是"我的邮 箱地址"。当单击该链接,系统将发送邮件给"my@mysample.com"。

   在实现以上控件之前,首先应分析如下:本控件包含外观元素,例如,文字大小、颜色、是否粗体等。为此,控件基类不应从Control类继承,而应从WebControl类继承。这样,开发人员将不必自行实现这些外观样式属性等内容。

   下面列举了实现自定义服务器控件的RenderContentsControl.cs文件源代码。

 using System;
using System.ComponentModel;
using System.Security.Permissions;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;

namespace CustomControlTest2.App_Code
{
    [
       AspNetHostingPermission(SecurityAction.Demand, Level = AspNetHostingPermissionLevel.Minimal),
       AspNetHostingPermission(SecurityAction.InheritanceDemand, Level = AspNetHostingPermissionLevel.Minimal),
        DefaultProperty("Email"),
        ParseChildren(true, "Text"),
       ToolboxData("<{0}:RenderContentsControl runat=server></{0}:RenderContentsControl>")
    ]
    public class RenderContentsControl : WebControl
    {
        // 实现Email属性
        [Bindable(true)]
        [Category("Appearance")]
        [DefaultValue("")]
        [Localizable(true)]
        public string Email
        {
            get
            {
                String s = (String)ViewState["Email"];
                return ((s == null) ? String.Empty : s);
            }
            set
            {
                ViewState["Email"] = value;
            }
        }
        // 实现Text属性
        [
         Bindable(true), Category("Appearance"), DefaultValue(""), Localizable(true),
         PersistenceMode(PersistenceMode.InnerDefaultProperty)
        ]
        public virtual string Text
        {
            get
            {
                string s = (string)ViewState["Text"];
                return (s == null) ? String.Empty : s;
            }
            set
            {
                ViewState["Text"] = value;
            }
        }
        // 重写TagKey属性
        protected override HtmlTextWriterTag TagKey
        {
            get
            {
                return HtmlTextWriterTag.A;
            }
        }
        // 重写AddAttributesToRender方法
        protected override void AddAttributesToRender(HtmlTextWriter writer)
        {
            base.AddAttributesToRender(writer);
            writer.AddAttribute(HtmlTextWriterAttribute.Href, "mailto:" + Email);
        }
        // 重写RenderContents方法
        protected override void RenderContents(HtmlTextWriter writer)
        {
            if (Text == String.Empty)
            {
                Text = Email;
            }
            writer.WriteEncodedText(Text);
        }
    }
}


 如 上代码所示,RenderContentsControl类继承自WebControl基类,其原因在前文已经说明。另外,在 RenderContentsControl类的实现过程中,还包括了元数据属性标记、3个属性(Email、Text和TagKey)和2两个方法 (AddAttributesToRender和RenderContents)实现等内容。下面逐一对这些内容进行分析。

   代码说明之3个属性:

   在上文代码中主要包括了3个属性:Email、Text和TagKey。Email属性用于获取或者设置具体的电子邮件地址,Text属性用于获取或者设置控件显示的文本内容。在这两个属性实现中,都使用了控件视图状态ViewState。服务器控件的视图状态为其所有属性值的累计。对于简单属性的实现,将经常使用ViewState。TagKey属性是重写属性,其继承自WebControl基类,这是读者需要理解的重点内容。重写TagKey属性主要是为了呈现HTML标记中的a元素,这样就不会呈现WebControl类所默认呈现的span元素此处,也暗示了本控件使用的构造函数是继承自WebControl的protected WebControl (),读者可返回上文再看看有关这个构造函数的说明。需要读者牢记的是:如果要呈现的元素是HtmlTextWriterTag枚举的成员,则应重写TagKey属性。 许多常见的HTML元素标记被映射为HtmlTextWriterTag枚举的值。例 如,System.Web.UI.HtmlTextWriterTag.A与a元素对应,而 System.Web.UI.HtmlTextWriterTag.Table与table元素对应。如果要呈现的元素不是由HtmlTextWriterTag枚举的成员表示,那么建议重写TagName属性,并返回要作为元素呈现的字符串。

   代码说明之2个方法:

   在RenderContentsControl服务器控件中重写了2个重要方法:一个是AddAttributesToRender、另一个是RenderContents。

   (1)AddAttributesToRender

 该方法用于为控件添加一个Href属性,并将该属性值设置为"mailto:Email",其中Email是上文所述的表示邮件地址的属性。当重写AddAttributesToRender方法时,应始终按照控件源代码演示的方式:首先,调用基类方法,然后进行相关设置。这样才能实现为服务器控件添加样式和其他属性的功能。实际上,根据前文所述的RenderBeginTag方法的实现示意性代码可知,AddAttributesToRender方法是由WebControl的RenderBeginTag方法调用。

   (2)RenderContents

   该方法是本示例的核心内容,其用于在控件的标记中写入由Text属性指定的超链接文本。如代码所示,服务器控件调用了HtmlTextWriter实例的WriteEncodedText方法,以对开发人员输入的文本进行HTML编码。一般情况下,为了安全起见,应该对用户提供的文本进行HTML编码。

   代码说明之元数据属性标记:

   元数据属性标记在类前和属性实现前都有所应用。有关这些元数据属性标记的说明,在以前的文章中已经进行了具体说明。读者可查阅有关讲解如何创建一个简单的服务器控件的文章。下面重点说明一下有关内部文本持久性的问题

   在类前的元数据属性标记中,设置了ParseChildren(true, "Text")。在Text属性前的元数据属性标记中,设置了PersistenceMode(PersistenceMode.InnerDefaultProperty)。通过以上两个设置,则为控件添加了内部文本持久性设置。如此设置,可使得开发人员能够在控件标记代码内设置Text属性。例如:

 <Sample:RenderContentsControl runat="server" ID="CustomerControl" Email="my@mysample.com">我的邮箱地址</Sample:RenderContentsControl>



   在上面的代码中,原来文本"我的邮箱地址"应由Text属性设置(这就是默认持久性),然而,由于内部文本持久性设置,因此,可以在控件标记内设置控件文本。



 实现内部持久性,应使用ParseChildren(true, "Text")来标记RenderContentsControl控件。ParseChildren的第一个参数true,其用于指定页分析器应将控件标记内的内容分析为属性,而不是子控件。第二个参数提供控件的内部默认属性名称为Text。使用这两个参数调用ParseChildren构造函数时,控件标记内的内容必须与内部默认属性对应。Text属性的PersistenceMode(PersistenceMode.InnerDefaultProperty),用于指定可视化设计器应将此属性作为控件标记中的内部内容进行序列化。

   通常,WebControl类使用PersistChildren(false)和ParseChildren(true)控制设计时和分析时属性的持久性。这两个属性将被控件继承,且仅在要更改继承的设置时需要应用。PersistChildren告知设计器是否应将服务器控件的子控件作为嵌套的内部控件保存false参数指示内部内容与属性对应,而不是与子控件对应。ParseChildren已在上一段中加以说明。如果WebControl类的设计时和分析时持久性适用于您的控件,则不必重写从WebControl继承的PersistChildren和ParseChildren。

   下面列举了用于测试服务器控件的Default.aspx文件源代码:

 <%@ Page Language="C#" AutoEventWireup="true" CodeFile="Default.aspx.cs" Inherits="_Default" %>
 <%@ Register TagPrefix="Sample" Assembly="UsingRenderContentsControl" Namespace="UsingRenderContentsControl" %>
 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
 <html xmlns="http://www.w3.org/1999/xhtml">
 <head id="Head1" runat="server">
 <title>使用RenderContents方法实现控件呈现</title>
 </head>
 <body>
 <form id="form1" runat="server">
 <div>
 <Sample:RenderContentsControl runat="server" ID="CustomerControl" Font-Bold="true" Font-Size="small" ForeColor="Blue" Email="my@mysample.com">我的邮箱地址</Sample:RenderContentsControl>
 </div>
 </form>
 </body>
 </html>



 如 上粗体代码所示,RenderContentsControl控件中设置了Font-Bold、Font-Size、ForeColor、Email等属 性,同时,还在控件标记之间设置了文本内容。当然,如果开发人员将Text属性值设置为相同的文本内容也是可以的。以上代码比较简单,在此不再说明了。

    通过这个示例,我们需要重点掌握RenderContents和AddAttributesToRender方法,以及TagKey属性的使用。提请读 者注意的是虽然服务器控件的代码比较复杂,但是结构很简单。切不可被复杂的样式设置和客户端行为代码弄得不知所措,而是要注意体会两个重点方法的使用过 程。

   在服务器控件开发成功之后,最好能够查看其HTML代码,并将服务器代码和HTML代码作以比较,搞清楚每一条服务器控件代码 呈现了什么样的HTML代码。通过这个方法,相信读者能够对服务器控件的呈现方法有更加深刻的体会。下面列举了当用户在浏览器中运行以上页面,并查看相关 的Html源文件时可得到的代码:

 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
 <html xmlns="http://www.w3.org/1999/xhtml">
 <head id="Head1">
 <title> 使用RenderContents方法实现控件呈现</title>
 </head>
 <body>
 <form name="form1" method="post" action="Default.aspx" id="form1">
 <div>
 <input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE" value="/wEPDwULLTEyNTQwNjQyMDJkZOlJ3PyMGs2hmzn9MU6Ogt9V+5ag" />
 </div>
 <div>
 <a id="CustomerControl" href="mailto:my@mysample.com" style="color:Blue;font-size:Small;font-weight:bold;">我的邮箱地址</a>
 </div>
 </form>
 </body>
 </html>





    通过观察以上代码可知,自定义服务器控件RenderContentsControl实际呈现的结果是粗体所示部分的代码,其最终呈现为一个表示超链接 的<a>标记,其中包括href、Style和文本等属性值。它们的值与Default.aspx文件源代码 中,RenderContentsControl控件的属性设置有着密切关系。例如,Email属性值最终呈现为href属性值等等。读者可自行对照查 看,这对于理解控件呈现很有益处。

   小结

   本文主要介绍了WebControl类的一些基本知识,并且利用这些 基本知识创建了一个简单的控件呈现实例。实际上,从中读者应该能够总结出来,创建继承自WebControl基类的自定义服务器控件,其中最为重要的重写 RenderContents方法。记住这一点是非常重要的。至此,笔者已经利用两篇文章来介绍实现控件呈现的方法。在以后的文章中,将介绍有关为控件实 现属性等方面的内容。