2016-04-25 177 views
0

我希望用戶能夠通過單擊位於該UserControl內的Delete按鈕來刪除UserControl。UpdatePanel:UserControl內部的按鈕沒有觸發

btnAddChoice工作正常,但btnRemove在UserControl內部,並且btnRemove_Click未被觸發。

這裏是我的ShowChoices.aspx代碼:

<div>     
    <strong>Choices</strong> 
    <asp:UpdatePanel ID="UpdatePanel1" runat="server" ChildrenAsTriggers="true"> 
     <Triggers> 
      <asp:AsyncPostBackTrigger ControlID="btnAddChoice" EventName="Click" />                      
     </Triggers> 
     <ContentTemplate> 
      <ul class="list-unstyled">         
       <asp:PlaceHolder runat="server" ID="phChoices"> 
       </asp:PlaceHolder> 
      </ul> 
     </ContentTemplate> 
    </asp:UpdatePanel>    
</div> 

這裏是我的ShowChoices.aspx.cs代碼:

protected void btnAddChoice_Click(object sender, EventArgs e) 
{    
    Choice ctl = (Choice)LoadControl("~/Controls/Choice.ascx"); 
    ctl.ID = "choice" + PersistedControls.Count;     
    int j = PersistedControls.Count + 1; 
    ctl.SetSummary("Choice #" + j); 
    phChoices.Controls.Add(ctl); // the UserControl is added here 
    PersistedControls.Add(ctl); 

    AsyncPostBackTrigger trigger = new AsyncPostBackTrigger(); 
    trigger.ControlID = ctl.BtnRemoveUniqueID; // problem : BtnRemoveUniqueID = always null 
    trigger.EventName = "Click"; 
    UpdatePanel1.Triggers.Add(trigger); 
} 

在Choice.ascx:

<asp:Button ID="btnRemove" CssClass="btn-default" runat="server" Text="Remove this choice" CausesValidation="false" OnClick="btnRemove_Click"/> 

在選擇.ascx.cs

protected void btnRemove_Click(object sender, EventArgs e) 
{ 
    this.Parent.Controls.Remove(this); 
    List<Control> _persistedControls = (List<Control>) Session[Step2.PersistedControlsSessionKey]; 
    _persistedControls.Remove(this); 
    Session[Step2.PersistedControlsSessionKey] = _persistedControls; 
    UpdatePanel ctl = (UpdatePanel) this.Parent.FindControl("UpdatePanel1"); 
    if (ctl != null) 
    { 
     ctl.Update(); 
    } 
} 

回答

1

用戶控件創建一個單獨的命名容器,這意味着需要爲您更新面板額外的工作(這就是回答在這裏做得很好解釋它:Trigger an update of the UpdatePanel by a control that is in different ContentPlaceHolder

這是暴露一個按鈕的工作的完整示例在用戶控件內部,因此它可以充當觸發器並觸發父頁面中的事件。該按鈕使用子控件上的公共屬性公開,然後所有事件處理程序都附加在父項中。

Default.aspx的

<%@ Page Language="C#" AutoEventWireup="true" CodeFile="Default.aspx.cs" Inherits="_Default" %> 
<%@ Register Src="~/TestChildControl.ascx" TagName="Custom" TagPrefix="TestChildControl" %> 

<!DOCTYPE html> 

<html xmlns="http://www.w3.org/1999/xhtml"> 
<head runat="server"> 
    <title></title> 
</head> 
<body> 
    <form id="form1" runat="server"> 
     <asp:ScriptManager runat="server" /> 
     <div> 
      <asp:UpdatePanel runat="server" ID="updTest"> 
       <Triggers> 
        <asp:AsyncPostBackTrigger ControlID="btnAddControl" EventName="Click" /> 
       </Triggers> 
       <ContentTemplate> 
        <asp:Placeholder runat="server" ID="phControlContainer"></asp:Placeholder> 
       </ContentTemplate> 
      </asp:UpdatePanel> 
      <asp:Button runat="server" ID="btnAddControl" Text="Add Control" OnClick="btnAddControl_OnClick" /> 
     </div> 
    </form> 
</body> 
</html> 

Default.aspx.cs

注意,子控件必須在頁面加載在同一順序用相同的ID,以便爲重建事件正常開火。

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Web.UI; 
using System.Web.UI.WebControls; 

public partial class _Default : System.Web.UI.Page 
{ 
    protected void Page_Load(object sender, EventArgs e) 
    { 
     if (ControlIDs != null) 
     { 
      foreach (string controlID in ControlIDs) 
      { 
       AddChildControl(controlID); 
      } 
     } 
    } 

    protected void btnAddControl_OnClick(object sender, EventArgs e) 
    { 
     var rand = new Random(); 
     var controlID = string.Format("TestChildControl_{0}", rand.Next()); 

     AddChildControl(controlID); 
    } 

    protected void AddChildControl(string controlID) 
    { 
     TestChildControl childControl = (TestChildControl)LoadControl("~/TestChildControl.ascx"); 
     childControl.ID = controlID; 
     phControlContainer.Controls.Add(childControl); 

     childControl.RemoveControlButton.Click += btnRemoveControl_OnClick; 

     AsyncPostBackTrigger updateTrigger = new AsyncPostBackTrigger() { ControlID = childControl.RemoveControlButton.UniqueID, EventName = "click" }; 
     updTest.Triggers.Add(updateTrigger); 

     SaveControlIDs(); 
    } 

    private void SaveControlIDs() 
    { 
     ControlIDs = phControlContainer.Controls.Cast<Control>().Select(c => c.ID).ToList(); 
    } 

    protected void btnRemoveControl_OnClick(object sender, EventArgs e) 
    { 
     var removeButton = sender as Button; 

     if (removeButton == null) 
     { 
      return; 
     } 

     var controlID = removeButton.CommandArgument; 
     var parentControl = 
      phControlContainer.Controls.Cast<TestChildControl>().FirstOrDefault(c => c.ID.Equals(controlID)); 

     if (parentControl != null) 
     { 
      phControlContainer.Controls.Remove(parentControl); 
     } 

     SaveControlIDs(); 
    } 

    protected IEnumerable<string> ControlIDs 
    { 
     get 
     { 
      var ids = ViewState["ControlIDs"] ?? new List<string>(); 
      return (IEnumerable<string>) ids; 
     } 
     set { ViewState["ControlIDs"] = value; } 
    } 
} 

TestChildControl.ascx

<%@ Control Language="C#" AutoEventWireup="true" CodeFile="TestChildControl.ascx.cs" Inherits="TestChildControl" %> 
<div> 
    This is a test control 
</div> 
<div> 
    <asp:Button runat="server" ID="btnRemoveControl" Text="Remove Control" /> 
</div> 

TestChildControl.ascx.cs

請注意,在這裏我們揭露按鈕爲只讀屬性,所以我們可以將事件處理程序它並訪問其成員。爲方便起見,我將父控件ID分配爲CommandArgument。

using System; 
using System.Web.UI.WebControls; 

public partial class TestChildControl : System.Web.UI.UserControl 
{ 
    protected void Page_Load(object sender, EventArgs e) 
    { 
     btnRemoveControl.CommandArgument = this.ID; 
    } 

    public Button RemoveControlButton 
    { 
     get { return btnRemoveControl; } 
    } 
} 
+0

我在使用你的解決方案更新了我的問題,我嘗試可惜btnRemove總是空,顯然是因爲它尚未建立...... –

+0

@JohnMarston我添加了上述的全部工作的例子,說明我已經完成了這一點。 – DVK