2009-10-30 42 views
1

我在ASP.NET项目中出现了一堆这种样板代码。ASP.NET:用户控件可以访问它包装的控件

<div class="inputfield"> 
    <div class="tl"> 
    <span class="tr"><!-- --></span> 
    <span class="ll"><!-- --></span> 
    <div class="lr"> 
     <div class="cntnt"> 
     <asp:TextBox .../> 
     </div> 
    </div> 
    </div> 
</div> 

正如你可能已经猜到了,一切都在那个片段是除了最里面的文本字段纯样板。

在ASP.NET中避免这种样板文件的最佳方法是什么?在例如Django的我会做一个自定义的标签吧,因为这样的:

{% boiler %} 
<input ... /> 
{% endboiler %} 

我在想,也许我可以创建一个用户控件,但我已经找到了在ASP.NET用户控件的所有教程都非常简单和“自动关闭”,即他们不知道标签的内容。我需要的东西沿线︰

<Hello:MyControl> 
    <asp:TextBox .../> 
</Hello> 

所以我的问题是以下:什么是避免样板的最好方法?

回答

2

您可以使用ITemplate属性。因此,您可以在不同情况下注入不同的内容。

[PersistChildren(false), ParseChildren(true, "ContentTemplate")] 
public partial class WebUserControl1 : System.Web.UI.UserControl 
{ 

    [System.ComponentModel.Browsable(false), System.Web.UI.PersistenceMode(PersistenceMode.InnerProperty)] 
    public ITemplate ContentTemplate { get; set; } 

    protected override void CreateChildControls() 
    { 
     if (this.ContentTemplate != null) 
      this.ContentTemplate.InstantiateIn(this); 

     base.CreateChildControls(); 
    } 
} 
+0

对不起,但我真的不明白如何使用此代码段。在这种情况下,.ascx文件的外观如何? – 2009-11-02 13:08:13

+0

我想我现在明白了,但仍然有一个问题......如果我不想使用ContentTemplate或类似的东西,该怎么办? – 2009-11-02 14:45:38

+0

你为什么不想要那个? – 2009-11-02 17:11:25

0

将asp:TextBox与其他html标签一起放在用户控件中。提供匹配的文本框的属性,关于你的用户控件的属性,所以,你会做这样的事情:

<Hello:MyControl ID="myControl" runat="server" Width="300px" MaxLength="30" /> 

,然后宽度和最大长度的属性会一下就传送到内部文本框。

您还可以从usercontrol提供对文本框的访问权限,并在代码后面设置所有属性。

+0

这还不够,因为应该可以将任何内容放入控件中,而不仅仅是文本框。 – 2009-10-30 15:44:57

+0

啊 - 对不起,我错过了这部分的问题。 – 2009-10-30 16:16:10

+0

我现在意识到,我应该更清楚的那部分! :) – 2009-10-30 16:36:55

0

创建一个类是这样的:

[PersistChildren(false), ParseChildren(true, "ContentTemplate")] 
public class CustomContent:WebControl 
{ 
    [System.ComponentModel.Browsable(false), System.Web.UI.PersistenceMode(PersistenceMode.InnerDefaultProperty)] 
    public ITemplate ContentTemplate { get; set; } 

    private PlaceHolder m_placeHolder; 
    protected override void CreateChildControls() 
    { 
     m_placeHolder = new PlaceHolder(); 

     if (this.ContentTemplate != null) 
      this.ContentTemplate.InstantiateIn(m_placeHolder); 

     Controls.Add(m_placeHolder); 
    } 

    protected override void RenderContents(HtmlTextWriter writer) 
    { 
     writer.Write(@"<div class=""inputfield""> 
<div class=""tl""> 
<span class=""tr""><!-- --></span> 
<span class=""ll""><!-- --></span> 
<div class=""lr""> 
    <div class=""cntnt""> 
"); 
     base.RenderContents(writer); 

     writer.Write(@"  </div> 
</div> 
</div> 
</div> 
"); 
    } 

} 

该类不是一个“用户控制”这是一个“服务器控制”。你可以用用户控件做同样的事情,但是你会遇到设计师的问题。这将在设计师工作。 你可以把类似这样的标记在你的ASPX:

<uc1:CustomContent runat="server" ID="content"> 
     <asp:textbox runat="server"></asp:textbox> 
    </uc1:CustomContent> 

不要忘了在ASPX上方的注册页声明

<%@ Register tagprefix="uc1" Assembly="Assembly where CustomContent is" Namespace="namespace where CustomContent is" %> 

你可以把任何你想要的UC1内: CustomContent标签,它会在其周围呈现该样板html。如果您对ITemplate的工作方式感到好奇,那么msdn上有很多文章,等等。