3

我看到了很多关于这个话题的东西,但我找不到解决方案。我在一个事件中添加了许多下拉列表,但它们没有触发 SelectedIndexChanged 事件。这是 drplist 创建者代码:

foreach (var row in cmdSelectCats.ExecuteReader())
{
    var id = row["ProductCategoryID"].ToString();

    var dropDownStatus = new DropDownList {ID = "DrpStatus-" + id};

    dropDownStatus.Items.Add(new ListItem("Aktif", "1"));
    dropDownStatus.Items.Add(new ListItem("Pasif", "2"));

    dropDownStatus.AutoPostBack = true; 
    dropDownStatus.SelectedIndexChanged += Status_SelectedIndexChanged;

    var tableCell = new TableCell();
    tableCell.Controls.Add(dropDownStatus);
    dropDownStatus.SelectedValue = row["ProductCategoryStatusID"].ToString();
    tableRow.Cells.Add(tableCell);
    TblCatList.Rows.Add(tableRow);
}

当然还有我的活动:

public void Status_SelectedIndexChanged(object sender, EventArgs e)
{
    //DO SOMETHING
} 

我错过了什么?

4

3 回答 3

17

这是一个常见问题,它与页面生命周期有关:

看看以下问题:

按钮数组上的单击事件

点击事件后按钮数组消失

动态创建一个 ImageButton

现在创建动态控件时要记住的基本步骤是:

  • 当您不使用母版页时,应在PreInit事件中创建动态控件,如果是,则在Init事件中创建控件
  • 避免在这些事件中设置可以在每个帖子中更改的属性,因为当应用视图状态时(在帖子事件中),这些属性将被覆盖
  • 每次发布页面时都必须创建动态控件,避免这种情况if(!this.IsPostBack) this.CreatemyDynamicControls();
  • 当您在PreInitorInit事件中创建控件时,它们的状态将在发布事件中自动设置,这意味着LoadComplete即使您在每个帖子中再次创建它们,即使您没有明确设置,您的控件也会包含它们的状态他们的状态。请注意,当您处理在设计时创建的控件时,此行为是不同的,在这种情况下,已设置状态的事件是 Load 事件
  • 事件订阅应该在 之前发生,PageLoadComplete否则它们将不会被引发

考虑来自 MSDN 的以下描述

如果控件是在运行时动态创建或在数据绑定控件的模板中以声明方式创建的,则它们的事件最初不会与页面上其他控件的事件同步。例如,对于在运行时添加的控件,Init 和 Load 事件在页面生命周期中发生的时间可能比以声明方式创建的控件的相同事件要晚得多。因此,从它们被实例化的那一刻起,动态添加的控件和模板中的控件一个接一个地引发它们的事件,直到它们赶上它被添加到 Controls 集合中的事件。

以上对我来说不是很清楚,但我发现了以下内容。以下TextBox是在设计时创建的

    protected void Page_PreInit(object sender, EventArgs e)
    {
        this.txtDesignTextBox1.Text = "From PreInit";
        this.txtDesignTextBox1.Text += DateTime.Now.ToString();
    }

    protected void Page_Init(object sender, EventArgs e)
    {
        this.txtDesignTextBox2.Text = "From Init";
        this.txtDesignTextBox2.Text += DateTime.Now.ToString();
    }

    protected void Page_Load(object sender, EventArgs e)
    {
        this.txtDesignTextBox3.Text = "From Load";
        this.txtDesignTextBox3.Text += DateTime.Now.ToString();
    }

乍一看,您可能会认为在每篇文章中,所有文本框都使用当前日期进行更新,但事实并非如此,因为它们是在设计时创建的,它们严格遵循 ASP.Net 页面生命周期,这意味着它们的状态在 PreInit 和 Init 事件之后被覆盖,只有txtDesignTextBox3在每个帖子中更新,因为它的Text属性在设置视图状态后更新(在Load事件中)。

但是动态控件的行为是不同的,记住 MSDN 的描述:

对于在运行时添加的控件,Init 和 Load 事件可能会在页面生命周期的后期发生

考虑以下:

    protected void Page_PreInit(object sender, EventArgs e)
    {
        var textBox = new TextBox { Text = "From PreInit", Width = new Unit("100%") };
        textBox.Text += DateTime.Now.ToString();
        this.myPlaceHolder.Controls.Add(textBox);
    }

    protected void Page_Init(object sender, EventArgs e)
    {
        var textBox = new TextBox { Text = "From Init", Width = new Unit("100%") };
        textBox.Text += DateTime.Now.ToString();
        this.myPlaceHolder.Controls.Add(textBox);
    }

    protected void Page_Load(object sender, EventArgs e)
    {
        var textBox = new TextBox { Text = "From Load", Width = new Unit("100%") };
        textBox.Text += DateTime.Now.ToString();
        this.myPlaceHolder.Controls.Add(textBox);
    }

在这种情况下,控件的行为略有不同,在这种情况下,在每个帖子中,即使在事件中创建的控件也不会更新控件Load

原因是它们的生命周期事件发生在页面生命周期的后期,这意味着即使在 Load 事件之后它们的状态也会被覆盖

为了解决这个问题,你可以使用LoadComplete事件,在这个事件中你可以改变动态控件的状态:

    protected void Page_LoadComplete(object sender, EventArgs e)
    {
        var textBox = new TextBox { Text = "From LoadComplete", Width = new Unit("100%") };
        textBox.Text += DateTime.Now.ToString();
        this.myPlaceHolder.Controls.Add(textBox);
    }

在这种情况下,状态将在每个帖子中更新。

但是,考虑到您应该在事件之前订阅动态控件事件,LoadComplete否则它们将不会被引发。

...我知道我讨厌这种行为,这就是我喜欢 MVC 的原因

作为在设计时创建的控件的快速参考:注意LoadViewState方法是如何在and事件之后但在事件之前调用的。该事件被认为是稳定的,因为在此事件中您可以访问控件的视图状态。另请注意,该方法表示导致回发的控制事件,可以是,,,等,此事件在事件之后处理PreInitInitLoadLoadRaisePostBackEventSelectedIndexChangedClickLoad

在此处输入图像描述

有关完整的详细规范,请阅读MSDN 页面生命周期文档

于 2012-06-20T19:46:54.413 回答
1

我通常看到这是由页面生命周期问题引起的。如果控件仅在触发事件时创建,那么当您的索引更改事件触发时,控件不存在以将其绑定到回发。

例子:

  1. MyEvent 触发。下拉菜单已创建。指定的事件处理程序。
  2. 已触发索引更改事件。页面重新加载。未找到下拉菜单,无法触发。

您必须确保在 .NET 尝试处理事件之前创建下拉列表。

于 2012-06-20T18:51:06.450 回答
-4

您缺少:
1- 覆盖 SaveViewState
2- 覆盖 LoadViewState

我为这个问题提供了一个示例代码。我测试它。这是工作。

ASPX:

<form id="form1" runat="server">
<asp:Button ID="Button1" runat="server" Text="Button" OnClick="Button1_Click" />
<div id="myDiv" runat="server">
</div>
<asp:Label ID="lblDescription" runat="server"></asp:Label>
</form>

代码背后:

public partial class Default : System.Web.UI.Page
    {
        private List<string> values = new List<string>();

        protected void Page_Load(object sender, EventArgs e)
        {

        }        

        protected override object SaveViewState()
        {
            object baseState = base.SaveViewState();            
            object[] allStates = new object[2];
            allStates[0] = baseState;
            allStates[1] = values;
            return allStates;
        }

        protected override void LoadViewState(object savedState)
        {
            object[] myState = (object[])savedState;
            if (myState[0] != null)
                base.LoadViewState(myState[0]);
            if (myState[1] != null)
            {
                values = (List<string>)myState[1];
                MyRender();
            }
        }

        protected void Button1_Click(object sender, EventArgs e)
        {
            for (int i = 0; i < 10; i++)
            {
                DropDownList ddl = new DropDownList();
                ddl.ID = "ClientID" + i;

                ddl.Items.Add("Item 1");
                ddl.Items.Add("Item 2");
                ddl.AutoPostBack = true;                
                values.Add(ddl.SelectedValue);

                myDiv.Controls.Add(ddl);
            }
        }

        private void MyRender()
        {
            for (int i = 0; i < values.Count; i++)
            {
                DropDownList ddl = new DropDownList();
                ddl.ID = "ClientID" + i;

                ddl.Items.Add("Item 1");
                ddl.Items.Add("Item 2");
                ddl.AutoPostBack = true;
                ddl.SelectedIndexChanged += new EventHandler(ddl_SelectedIndexChanged);

                myDiv.Controls.Add(ddl);
            }
        }

        void ddl_SelectedIndexChanged(object sender, EventArgs e)
        {
            lblDescription.Text = ((DropDownList)sender).ID + ": Selected Index Changed";
        }
    }
于 2012-06-20T20:31:07.620 回答