自定义控件的构建(4)

    在自定义控件的构建(3)中,我们构建了组合控件,实际上,在大部分构建组合控件时,同时会指定其子控件的布局

构建混合控件

  下面的代码是设计为一个登录控件,同时指定其布局

 namespace MyCompositeControls
{    
    /// <summary>
    /// 构建混合控件
   /// </summary>
    public class Login : CompositeControl
    {
        private TextBox txtUserName;
        private TextBox txtPassword;
        public string UserName
        {
            get 
            {
                EnsureChildControls();
                return txtUserName.Text;
            }
            set
            {
                EnsureChildControls();
               txtUserName.Text=value;
            }
        }
        public string Password
        {
            get
            {
                EnsureChildControls();
                return txtPassword.Text;
            }
            set
            {
                EnsureChildControls();
                txtPassword.Text = value;
            }
        }
        protected override void CreateChildControls()
        {
            txtUserName = new TextBox();
            txtUserName.ID = "txtUserName";
            this.Controls.Add(txtUserName);
            txtPassword = new TextBox();
            txtPassword.ID = "txtPassword";
            this.Controls.Add(txtPassword);
        }
        protected override void RenderContents(HtmlTextWriter writer)
        {
            writer.RenderBeginTag(HtmlTextWriterTag.Tr);
            writer.RenderBeginTag(HtmlTextWriterTag.Td);
            writer.AddAttribute(HtmlTextWriterAttribute.For, txtUserName.ClientID);
            writer.RenderBeginTag(HtmlTextWriterTag.Label);
            writer.Write("User Name:");
            writer.RenderEndTag();
            writer.RenderEndTag();
            writer.RenderBeginTag(HtmlTextWriterTag.Td);
            txtUserName.RenderControl(writer);
            writer.RenderEndTag();
            writer.RenderEndTag();
            writer.RenderBeginTag(HtmlTextWriterTag.Tr);
            writer.RenderBeginTag(HtmlTextWriterTag.Td);
            writer.AddAttribute(HtmlTextWriterAttribute.For, txtPassword.ClientID);
            writer.RenderBeginTag(HtmlTextWriterTag.Label);
            writer.Write("Password:");
            writer.RenderEndTag();
            writer.RenderEndTag();
            writer.RenderBeginTag(HtmlTextWriterTag.Td);
            txtPassword.RenderControl(writer);
            writer.RenderEndTag();
            writer.RenderEndTag();
        }
        protected override HtmlTextWriterTag TagKey
        {
            get { return HtmlTextWriterTag.Table; }
        }
}
    }
 可以看到RenderContents()内部被重写为两个TextBox的布局顺序,TextBox是通过调用各自的RenderContents()生成的

运行后查看界面源码

 <table id="Login1">
  <tr>
    <td><label for="Login1_txtUserName">User Name:</label></td><td><input name="Login1$txtUserName" type="text" id="Login1_txtUserName" /></td>
  </tr><tr>
    <td><label for="Login1_txtPassword">Password:</label></td><td><input name="Login1$txtPassword" type="text" id="Login1_txtPassword" /></td>
  </tr>
</table>    

可见TextBox是生成在一个表格中的。

默认的RenderContents()只调用每个子控件的RenderContents(),通过重写控件的RenderContents()可以对子控件的布局进行更多的控制

下面的代码对RenderContents()进行了改写,实现<div>对控件进行布局的,因为在Web标准中,更支持对<div>的使用

 //重写布局
        protected override void RenderContents(HtmlTextWriter writer)
        {
            writer.AddStyleAttribute("float", "left");
            writer.RenderBeginTag(HtmlTextWriterTag.Div);
            writer.AddStyleAttribute(HtmlTextWriterStyle.Padding, "3px");
            writer.RenderBeginTag(HtmlTextWriterTag.Div);
            writer.AddAttribute(HtmlTextWriterAttribute.For, txtUserName.ClientID);
            writer.RenderBeginTag(HtmlTextWriterTag.Label);
            writer.Write("User Name:");
            writer.RenderEndTag();
            writer.RenderEndTag();       
           
            writer.AddStyleAttribute(HtmlTextWriterStyle.Padding, "3px");
            writer.RenderBeginTag(HtmlTextWriterTag.Div);
            writer.AddAttribute(HtmlTextWriterAttribute.For, txtPassword.ClientID);
            writer.RenderBeginTag(HtmlTextWriterTag.Label);
            writer.Write(" Password:");
            writer.RenderEndTag();
            writer.RenderEndTag();
            writer.RenderEndTag();
           
            writer.AddStyleAttribute("float", "left");
            writer.RenderBeginTag(HtmlTextWriterTag.Div);
            writer.AddStyleAttribute(HtmlTextWriterStyle.Padding, "3px");
            writer.RenderBeginTag(HtmlTextWriterTag.Div);
            txtUserName.RenderBeginTag(writer);        
            writer.RenderEndTag();
           
            writer.RenderBeginTag(HtmlTextWriterTag.Div);
            writer.AddStyleAttribute(HtmlTextWriterStyle.Padding, "3px");
            writer.RenderBeginTag(HtmlTextWriterTag.Div);
            txtPassword.RenderBeginTag(writer);
            writer.RenderEndTag();
            writer.RenderEndTag();
            writer.Write("<br style='clear:left'/>");
        }
        protected override HtmlTextWriterTag TagKey
        {
            get { return HtmlTextWriterTag.Div; }
        }
看看页面的Html生成
 
<div id="LoginStandard1">
  <div style="float:left;">
    <div style="padding:3px;">
      <label for="LoginStandard1_txtUserName">User Name:</label>
    </div><div style="padding:3px;">
      <label for="LoginStandard1_txtPassword"> Password:</label>
    </div>
  </div>
<div style="float:left;">
       <div style="padding:3px;">
      <input name="LoginStandard1$txtUserName" type="text" id="LoginStandard1_txtUserName" /><div>
        <div style="padding:3px;">
          <input name="LoginStandard1$txtPassword" type="text" id="LoginStandard1_txtPassword" />
        </div>
<br style='clear:left'/>
  </div>
尽管如此,在标准控件中Ms还是采用了Html表格布局方式。

 

 

本篇参考了《ASP.NET 3.5(揭秘) 卷2》

posted @ 2010-07-25 15:39  ringgo  阅读(741)  评论(0编辑  收藏  举报