2

我想在 asp.net 中使用中继器控件来实现 n 级数据层次结构。有什么解决方案可以实现这种层次结构吗?

4

2 回答 2

2

对于这个答案,我将建议以编程方式创建您的模板 - 请参见此处:https ://msdn.microsoft.com/en-us/library/aa289501 。可能有一些方法可以使用在标记中创建的模板,但这似乎更容易,而且肯定更灵活。

我从一个只有中继器(不是模板)的页面开始

<body>
    <form id="form1" runat="server">
    <div>
        <asp:Repeater runat="server" ID="TestRepeater">
        </asp:Repeater>
    </div>
    </form>     
</body>

和一个数据类

public class DataClass
{
    public string Name { get; set; }
    public List<DataClass> Children { get; set; }
}

对于模板,我们使用以下类:

public class DataTemplate : ITemplate
{
    public void InstantiateIn(Control container)
    {
        var name = new Literal();
        var repeater = new Repeater();

        name.DataBinding += BindingLiteral;
        repeater.DataBinding += BindingRepeater;

        // this here makes it recursive
        repeater.ItemTemplate = new DataTemplate();

        container.Controls.Add(name);
        container.Controls.Add(repeater);
    }

    private void BindingLiteral(object sender, System.EventArgs e)
    {
        var name = (Literal)sender;
        var container = (RepeaterItem)name.NamingContainer;
        name.Text = String.Concat("<h2>", DataBinder.Eval(container.DataItem, "Name").ToString(), "</h2>");
    }

    private void BindingRepeater(object sender, System.EventArgs e)
    {
        var name = (Repeater)sender;
        var container = (RepeaterItem)name.NamingContainer;
        name.DataSource = DataBinder.Eval(container.DataItem, "Children");
    }
}

显然,您会想要使用更复杂的模板。请注意,如果您当前在标记中有一个模板,您可以简单地获取由标记解析器生成的代码,并根据您的需要对其进行调整。

现在在页面后面的代码中,我们简单地分配 ItemTemplate 和 DataSource:

public partial class Test : System.Web.UI.Page
{
    protected void Page_Init(object sender, EventArgs e)
    {
        TestRepeater.DataSource = GetTestData();
        TestRepeater.ItemTemplate = new DataTemplate();
        TestRepeater.DataBind();
    }
}

这样做的好处是您的模板只是一个类,因此您可以添加一个类public Int32 Depth { get; set; },并根据您的深度更改生成的控件。

于 2015-04-01T11:58:09.787 回答
1

另一种解决方案,无需以编程方式创建模板:

使用一个简单的数据类:

public class DataClass
{
    public string Name { get; set; }
    public List<DataClass> Children { get; set; }
}

在 ASPX 标记中创建您的父转发器,将您的项目显示代码放入 ItemTemplate,并添加第二个“空”转发器:

<body>
    <form id="form1" runat="server">
    <div>
        <asp:Repeater runat="server" ID="ParentRepeater" OnItemDataBound="Repeater_ItemDataBound">
            <ItemTemplate>
                <asp:Literal runat="server" Text="<%# Eval("Name") %>"></asp:Literal>
                <asp:Repeater runat="server" ID="ChildRepeater" OnItemDataBound="Repeater_ItemDataBound" Visible="false">
                </asp:Repeater>
            </ItemTemplate>
        </asp:Repeater>
    </div>
    </form>     
</body>

在代码隐藏中:

protected void Repeater_ItemDataBound(object sender, RepeaterItemEventArgs e) {
    if (e.Item.ItemType == ListItemType.Item || e.Item.ItemType == ListItemType.AlternatingItem) {
        DataClass currentItem = (DataClass)e.Item.DataItem;

        if (currentItem.Children.Count > 0) {
            Repeater ChildRepeater = (Repeater)e.Item.FindControl("ChildRepeater");

            ChildRepeater.DataSource = currentItem.Children;
            ChildRepeater.ItemTemplate = ParentRepeater.ItemTemplate;
            ChildRepeater.Visible = true;
            ChildRepeater.DataBind();
        }
    }
}
于 2018-07-31T13:56:41.263 回答