将ASP.NET用户控件转化为自定义控件

作者:Kevin Cheng (程建和) 
最后修改时间:2006-03-14 
概述:如何将ASP.NET用户控件移植为ASP.NET自定义控件 
关键字:Asp.net, 用户控件, 自定义控件 
本文适用读者: 
   - 熟悉aspnet,能创建ascx用户控件 
   - 想创建自定义控件,而又为其庞杂的实现方法而惧的读者 
相关下载: 
   - 本文使用的ascx控件例程:WebFtp060308(pm06).rar 
   - 本文使用的自定义控件例程:WebFtpControl 1.0.2259_source.rar 
术语声明: 
    Field       类成员变量 
    Property    类属性 
    Attribute   类特性 
    NameSpace   命名空间 
    Assembly  程序集

引言:  
    asp.net自定义控件使用起来相当方便,但相对ascx用户控件而言,由于其没有可视化编程支持,编写时相当困难。 
    有没有简单的办法呢?既可以使用ascx可视化编程的优点,又可以发挥自定义控件使用方便的特点。 
    本文介绍一种解决方案:先设计好ascx控件,再转化为用户控件。 以下为具体步骤。 
     
    
一、创建ascx控件,并调试通过  
    
    具体怎么创建ascx用户控件不是本文的重点,跳过啦~ 
    本文使用到的ascx例程可点击此处下载。该例程创建了一个web文件管理的ascx控件。效果如: 
        

二、将ascx移植为自定义控件

(一)新建工程aspnet自定义控件工程,将原ascx类中的所有cs代码都拷贝到自定义控件类中去 
    (二)修改InitializeComponent()函数以初始化控件。可将其扩充为三个小函数,如:

private void InitializeComponent()   
{   
    CreateComponent();   
    SetComponentProperty();   
    SetComponentEvent();   
}   
其中   
(1)CreateComponent()函数用于新建出所有使用到的控件,并组织好其层次关系   
    private void CreateComponent()   
    {   
        // create   
        lblCurrPath = new Label();   
        lblUpload = new Label();   
        lblNewFolder = new Label();   
         
           
        // nest   
        Controls.Add(this.lblCurrPath);   
        Controls.Add(new HtmlGenericControl("br"));   
        this.panAdmin.Controls.Add(this.file);   
         
    }   
(2)SetComponentProperty()函数用于设置这些控件的属性,如   
    private void SetComponentProperty()   
    {   
        lblCurrPath.Font.Size = FontUnit.XLarge;   
        lblCurrPath.CssClass = "webftp_title";   
        panInfo.BackColor = Color.Yellow;   
        panInfo.Visible = false;   
         
    } 
    值得注意的是,aspnet提供的组合绑定控件(如DataGrid, DataList..)的属性设置较为复杂,参见附录二  
(3)SetComponentEvent()函数中设置控件的事件句柄,这个从原有的InitializeComponent函数中拷贝过来就可,如  
private void SetComponentEvent()   
{   
    this.btnUpload.Click += new System.EventHandler(this.btnUpload_Click);   
    this.btnNewFolder.Click += new System.EventHandler(this.btnNewFolder_Click);   
    this.dg.ItemCommand += new System.Web.UI.WebControls.DataGridCommandEventHandler(this.dg_ItemCommand);   
    this.Load += new System.EventHandler(this.Page_Load);   
     
}  

(三)填充Render()函数用以渲染输出效果。可将其扩展为两个小函数,如

protected override void Render(HtmlTextWriter output)  
{  
    RenderClientScript(output);  
    RenderServerComponent(output);  
}  
其中  
(1)RenderClientScript函数可用于输出客户端脚本,以及样式style代码,如:  
    private void RenderClientScript(HtmlTextWriter output)  
    {  
        output.Write("<style>{0} {1}</style>", "A { TEXT-DECORATION: none }", "A:hover { TEXT-DECORATION: underline }");  
          
    }  
(2)RenderServerComponent函数用于输出各控件,也可在此添加一些普通html代码的输出,如  
    private void RenderServerComponent(HtmlTextWriter output)  
    {  
        this.lblCurrPath.RenderControl(output);  
        output.Write("<hr width=100% size=1 color=darkgray>");  
        this.panAdmin.RenderControl(output);  
        this.dg.RenderControl(output);  
    } 

(四)让类实现INamingContainer接口,以避免页面上放置多个该控件而导致子控件id冲突的情况 
        该接口无任何函数,只需声明实现就可,如

public class WebFtp : System.Web.UI.WebControls.WebControl, INamingContainer  

只要让类实现该接口,系统将自动实现避免id冲突的方法 
        具体步骤是将内部控件id名前都附上父控件的id。如在panel(id=pan)中有一个lable(id=lblName),那么label的id会给自动改为pan_lblName; 
     至此,编译调试成功后,该控件就可以使用了。再加上图片,效果可以是这样: 
        

三、商业标准化自定义控件

第二步完成后,该控件已经可以使用了(如何使用?晕,请看附录五),但其还是一个粗胚,例如没有控件图标,没有设计时外观等。 
    以下将介绍如何修饰该控件,令其像个标准化的商业控件。 
    (1)为控件的属性添加例如类别、描述、默认值等Attribute特性 
        这些特性将影响到控件在属性工具栏中的内容 
        格式如:[Bindable(true), Category("类别"), Description("描述信息"), DefaultValue("缺省字符串类型值")] 
        常用的Attribute有:

CategoryAttribute               属性类别  
BrowsableAttribute              属性可否显示  
DefaultPropertyAttribute        控件的缺省属性  
DefaultValueAttribute           属性的缺省值  
DescriptionAttribute            属性的描述信息  
ReadOnlyAttribute               属性是否只读  
BindableAttribute               属性是否可绑定  
NotifyParentPropertyAttribute   修改该属性后是否通知控件  

更多的Attribute 请查阅附录三、附录四、以及 System.ComponentModel 命名空间 
    (2)添加工具栏图标 
        为类添加特性:[ToolboxBitmap(typeof(WebFtp), "Kingsoc.Web.Controls.WebFtp.bmp")] 
        其中要注意的是: 
        - 项目的默认命名空间为Kingsoc.Web.Controls 
        - 图片名称为webftp.bmp,并需要设置其编译选项为“嵌入的资源” 
        这样工具栏上图标就变成这样咯~: 
    (3)添加设计时外观支持 
        为类添加特性:[Designer(typeof(Kingsoc.Web.Controls.WebFtpDesigner))] 
        其中,类 Kingsoc.Web.Controls.WebFtpDesigner 继承至ControlDesigner,用于绘制设计时的控件外观,如:

        using System;  
        using System.Web.UI;  
        using System.Web.UI.WebControls;  
        using System.ComponentModel;  
        using System.ComponentModel.Design;  
        using System.Web.UI.Design;  
        using System.Text;  
        namespace Kingsoc.Web.Controls  
        {  
         public class WebFtpDesigner : ControlDesigner  
         {  
                public override string GetDesignTimeHtml()  
                {  
                    //return base.GetDesignTimeHtml ();  
                    WebControl ctrl = (WebControl)this.Component;  
                    StringBuilder sb = new StringBuilder();  
                    sb.AppendFormat(  
                        "<label style='align:center; valign:middle; font-size:small; font-family: Arial; #eeeeee; padding:2px; border:2px dotted gray; width:{0}; height:{1};'><center>{2}</center></lable>",  
                        ctrl.Width,  
                        ctrl.Height,  
                        ctrl.ID);  
                    return sb.ToString();  
                }  
         
                public override bool AllowResize  
                {  
                    get {return true;}  
                }  
         }  
        } 

这样,设计时控件就有了外观支持:。 
        (嫌太简陋?自己用dreamweaver等设计工具设计,想多好看就多好看!再拷贝html代码到GetDesignTimeHtml()函数内部去) 
     (4)添加TagPrefix 
        用于指定将控件从工具栏中拖放到窗体中,自动生成的控件名称前缀(实际上是控件所属的命名空间的别名) 
        在AssemblyInfo.cs文件中添加:[assembly: TagPrefix("Kingsoc.Web.Controls", "ks")] 
        这样,添加了该控件的web窗体就会自动生成类似代码

<%@ Register TagPrefix="ks" Namespace="Kingsoc.Web.Controls" Assembly="Kingsoc.Web.Controls.WebFtp" %> 
<ks:WebFtp id="webFtp" runat="server"></ks:WebFtp>

(5)设置版权信息 
        修改AssemblyInfo.cs文件中的以下特性

[assembly: AssemblyTitle("")]  
[assembly: AssemblyDescription("")]  
[assembly: AssemblyConfiguration("")]  
[assembly: AssemblyCompany("Kingsoc software")]  
[assembly: AssemblyProduct("")]  
[assembly: AssemblyCopyright("")]  
[assembly: AssemblyTrademark("Kingsoc")]  
[assembly: AssemblyCulture("")] 

(6)将控件添加到GAC中。见附件五 
     (7)添加控件注册保护。见附件六

--------------------------------------------------------------------        
附录 
--------------------------------------------------------------------     
附录一:自定义控件基类的选择  
    Control     所有web控件的基类 
    WebControl  继承至Control类,并添加了例如Width,Height等属性,这是常用且默认的基类,常用于有外观的控件 
    Component   常用于无外观的控件,例如:数据访问控件 
    故,对于组合控件,一般使用WebControl作为基类。而对于无界面的控件,一般使用Component就可以了

附录二:用代码创建DataGrid绑定列  
    (1)普通绑定列

BoundColumn col = new BoundColumn();  
col.DataField = "name";  
col.ReadOnly = true;  
dg.Columns.Add(col); 

(2)模板绑定列

TemplateColumn col = new TemplateColumn();  
col.ItemTemplate = new LinkButtonTemplate("btnName", "name", "Select");  
col.EditItemTemplate = new TextBoxTemplate("txtName", "name");  
dg.Columns.Add(col);  
其中LinkButtonTemplate和TextBoxTemplate模板代码如下:  
// LinkButton模板  
public class LinkButtonTemplate : ITemplate  
{  
    string _id;  
    string _bindField;  
    string _commandName;

public LinkButtonTemplate(string id, string bindField, string commandName)  
    {  
        _id = id;  
        _commandName = commandName;  
        _bindField = bindField;  
    }  
    public void InstantiateIn(Control container)  
    {  
        LinkButton ctrl = new LinkButton();  
        ctrl.ID = _id;  
        ctrl.CommandName = _commandName;  
        ctrl.DataBinding += new EventHandler(this.BindData);  
        container.Controls.Add(ctrl);  
    }  
    void BindData(object sender, System.EventArgs e)  
    {  
        LinkButton ctrl = sender as LinkButton;  
        ctrl.Text = DataBinder.Eval(ctrl.NamingContainer, "DataItem." + _bindField).ToString();  
        //或 ctrl.Text = ((ctrl.NamingContainer as DataGridItem).DataItem as DataRowView)[_bindField].ToString();  
    }  
}

// TextBox模板  
public class TextBoxTemplate : ITemplate  
{  
    string _id;  
    string _bindField;

public TextBoxTemplate(string id, string bindField)  
    {  
        _id = id;  
        _bindField = bindField;  
    }  
    public void InstantiateIn(Control container)  
    {  
        TextBox ctrl = new TextBox();  
        ctrl.ID = _id;  
        ctrl.DataBinding += new EventHandler(this.BindData);  
        container.Controls.Add(ctrl);  
    }  
    void BindData(object sender, System.EventArgs e)  
    {  
        TextBox ctrl = sender as TextBox;  
        ctrl.Text = DataBinder.Eval(ctrl.NamingContainer, "DataItem." + _bindField).ToString();  
    }  
}

附录三:自定义控件可用的特性(Attribute)  
    System.ComponentModel 命名空间提供用于实现组件和控件运行时和设计时行为的类。 
    此命名空间包括用于实现属性和类型转换器、绑定到数据源以及授权组件的基类和接口。 
        CategoryAttribute 
        BrowsableAttribute 
        DefaultPropertyAttribute 
        DefaultValueAttribute 
        DescriptionAttribute 
        ReadOnlyAttribute 
        NotifyParentPropertyAttribute 
        
        AmbientValueAttribute 
        BindableAttribute 
        DefaultEventAttribute 
        DesignerAttribute 
        DesignerCategoryAttribute 
        DesignerSerializationVisibilityAttribute 
        DesignOnlyAttribute 
        DesignTimeVisibleAttribute 
        EditorBrowsableAttribute 
        ImmutableObjectAttribute 
        InheritanceAttribute 
        InstallerTypeAttribute 
        LicenseProviderAttribute 
        ListBindableAttribute 
        MergablePropertyAttribute 
        ParenthesizePropertyNameAttribute 
        ProvidePropertyAttribute 
        RecommendedAsConfigurableAttribute 
        RefreshPropertiesAttribute 
        RunInstallerAttribute 
        ToolboxItemAttribute 
        ToolboxItemFilterAttribute 
        TypeConverterAttribute  
     
    
附录四:关于DefaultValue特性 
    只用于检测用户输入值是否与该值相同,若相同,则不会在asp页面代码中显示出该属性值;反之则显示。实际并没有修改该属性的值! 
    例如,对于以下语句,AttachTimeStampToUploadedFile 应该等于构造函数中指定的true,而非DefaultValue中提供的false

[DefaultValue(false)]  
public bool AttachTimeStampToUploadedFile  
{  
    get {return _AttachTimeStampToUploadedFile;}  
    set {_AttachTimeStampToUploadedFile = value;}  
}  
public WebFtp()  
{  
    AttachTimeStampToUploadedFile = true;  

所以,设置默认值时应该确保与初始化值相同,否则会带来令人困惑的结果。要么干脆就不写。

附录五:如何将自定义控件添加到工具栏  
    (1)在工具栏面板中点击右键,点击”添加/移除项..." 
    (2)在弹出的“自定义工具箱”对话框中点击“浏览”按钮,选择自定义控件dll文件 
    (3)点击确定,关闭对话框。控件图标将显示在工具栏面板上

附录六:将控件添加到GAC中  
    .Net framework 要求组件必须是 Strong-Name Assembly类型才能部署到GAC(全局程序集缓存Global Assembly Cache)。将组件部署到GAC中的好处就像是以往把Win32 DLL放置到System32目录中的效果,所有程序都可以共享这个组件,而无需再每个程序的目录中都放置一个assembly,这样可以减少部署的文件数和大小。
    (1) 首先先生成keyfile,可以用sn.EXE工具(在VS安装目录\SDK\v1.1\Bin下可找到)生成:
       sn -k MyCompany.snk
    (2) 把keyfile放置到项目根目录下,并在AssemblyInfo.cs中指定这个keyfile,并重新编译
       [assembly: AssemblyKeyFile(@"..\..\MyCompany.keys")]
    (3) 部署到GAC中:管理工具->Microsoft.Net Framework 1.1 配置->任务"向程序集缓存中添加程序集",指定编译好的dll。也可以直接使用GACUtil:
     - 把程序集添加到GAC中: GACUtil /i sample.dll  
     - 把程序集移出GAC:GACUtil /u sample.dll

将ASP.NET用户控件转化为自定义控件的更多相关文章

  1. 033. asp.netWeb用户控件之二将页面转换成web控件和使用Web控件显示热点新闻

    访问Web用户控件的属性 ASP.NET提供的各种服务器控件都有其自身的属性和方法,程序开发人员可以灵活地使用服务器控件中的属性和方法开发程序.在用户控件中,程序开发人员也可以自行定义各种属性和方法, ...

  2. 039. asp.netWeb用户控件之七实现具有虚拟键盘的功能的用户控件

    用户控件ascx代码: <%@ Control Language="C#" AutoEventWireup="true" CodeFile="K ...

  3. 038. asp.netWeb用户控件之六实现日期选择的用户控件

    web用户控件的ascx代码: <%@ Control Language="C#" AutoEventWireup="true" CodeFile=&qu ...

  4. 037. asp.netWeb用户控件之五使用用户控件实现文件上传功能

    fileUpload.ascx代码: <%@ Control Language="C#" AutoEventWireup="true" CodeFile= ...

  5. 036. asp.netWeb用户控件之五使用用户控件实现分页数据导航

    UserDataPager.ascx用户控件代码: <%@ Control Language="C#" AutoEventWireup="true" Co ...

  6. 035. asp.netWeb用户控件之四通过用户控件实现投票和结果分析

    用户控件Vote.ascx代码 <%@ Control Language="C#" AutoEventWireup="true" CodeFile=&qu ...

  7. 034. asp.netWeb用户控件之三通过用户控件实现用户注册和登录

    用户控件login.ascx代码: <%@ Control Language="C#" AutoEventWireup="true" CodeFile=& ...

  8. ASP.NET用户控件事件的定义和实践

    假定用户控件(UserControl.ascx)中包含按钮控件  AButton,希望实现按  Button  按钮时,包含该用户控件的页面可以接收到事件. UserControl.ascx.cs   ...

  9. ASP.NET - 用户控件制作

    首先添加用户控件: 在里面写上代码: <%@ Control Language="C#" AutoEventWireup="true" CodeBehin ...

随机推荐

  1. 启明星会议室系统与Office365集成说明

    在本文,我们将介绍如何配置Office365,以便改系统能够支持启明星会议室预定系统. In this article, we will introduct how to config microso ...

  2. CubieBoard 简单入门

    大约一个月之前折腾的部分记录,当时没有完全完成,就着手其他事情了,这是存在Live Writer中的草稿,先发出来吧,后来花了一段时间移植Qt,一直遇到了点问题,并没有完全跑通,后续估计也没有时间再继 ...

  3. [2014亚马逊amazon] 在线笔试题 大于非负整数N的第一个回文数 Symmetric Number

    1.题目 如标题,求大于整数N(N>=0)的第一个回文数的字符串表示形式. 这个题目也是当时笔试第一次见到,花了一个小时才做出了.慢慢总结还是挺简单的. 2.分析 分析如下: (1)一位数N(9 ...

  4. 解决在非Activity中使用startActivity

    错误提示信息: Caused by: android.util.AndroidRuntimeException: Calling startActivity() from outside of an ...

  5. python机器学习sklearn 岭回归(Ridge、RidgeCV)

    1.介绍 Ridge 回归通过对系数的大小施加惩罚来解决 普通最小二乘法 的一些问题. 岭系数最小化的是带罚项的残差平方和, 其中,α≥0α≥0 是控制系数收缩量的复杂性参数: αα 的值越大,收缩量 ...

  6. redis调优 -- 内存碎片

    最近查看了一下redis运行状况,发现公司测试服务器的redis内存不太够用,但是实际占用内存的数据量其实不大,以前也没有这种情况,之前在cache层新增了一个防刷积分任务的逻辑才会这样,搜索一下原因 ...

  7. Android -- 获取视频第一帧缩略图

    干货 从API 8开始,新增了一个类: android.media.ThumbnailUtils这个类提供了3个静态方法一个用来获取视频第一帧得到的Bitmap,2个对图片进行缩略处理. public ...

  8. 转:fastText原理及实践(达观数据王江)

    http://www.52nlp.cn/fasttext 1条回复 本文首先会介绍一些预备知识,比如softmax.ngram等,然后简单介绍word2vec原理,之后来讲解fastText的原理,并 ...

  9. 用GibbsLDA做Topic Modeling

    http://weblab.com.cityu.edu.hk/blog/luheng/2011/06/24/%E7%94%A8gibbslda%E5%81%9Atopic-modeling/#comm ...

  10. How to Sign in as a Different User in SharePoint 2013

    SharePoint used to have a menu option called "Sign in as Different User" in the top-right ...