在用户控件中,我在UpdatePanel内有一个Repeater(此ID在ModalPopupExtender内显示。Repeater使用MyDTO对象的数组列表进行数据绑定(bind)。列表中的每个Item都有两个按钮。设置了ImageURL和CommandArgument。
这段代码在第一次就可以正常工作,但是此后CommandArgument是错误的。似乎显示已正确更新,但DTO尚未更新,并且发送的CommandArgument是刚刚被删除的CommandArgument。
有人能发现代码有问题吗?
编辑:我刚刚在代码中添加了CollapsiblePanelExtender。现在,当我删除一个项目并展开面板时,以前被删除(并从显示中删除)的项目又回来了。中继器似乎没有在阀盖下正确重建。
ASCX
<asp:UpdatePanel ID="ViewDataDetail" runat="server" ChildrenAsTriggers="true">
<Triggers>
<asp:PostBackTrigger ControlID="ViewDataCloseButton" />
<asp:AsyncPostBackTrigger ControlID="DataRepeater" />
</Triggers>
<ContentTemplate>
<table width="100%" id="DataResults">
<asp:Repeater ID="DataRepeater" runat="server" OnItemCommand="DataRepeater_ItemCommand" OnItemDataBound="DataRepeater_ItemDataBound">
<HeaderTemplate>
<tr>
<th><b>Name</b></th>
<th><b> </b></th>
</tr>
</HeaderTemplate>
<ItemTemplate>
<tr>
<td>
<b><%#((MyDTO)Container.DataItem).Name%></b>
</td>
<td>
<asp:ImageButton CausesValidation="false" ID="DeleteData" CommandName="Delete" runat="server" />
<asp:ImageButton CausesValidation="false" ID="RunData" CommandName="Run" runat="server" />
</td>
</tr>
<tr>
<td colspan="2">
<table>
<tr>
<td>Description : </td>
<td><%#((MyDTO)Container.DataItem).Description%></td>
</tr>
<tr>
<td>Search Text : </td>
<td><%#((MyDTO)Container.DataItem).Text%></td>
</tr>
</table>
</td>
</tr>
</ItemTemplate>
</asp:Repeater>
</table>
</ContentTemplate>
</asp:UpdatePanel>
代码隐藏
public DeleteData DeleteDataDelegate;
public RetrieveData PopulateDataDelegate;
public delegate ArrayList RetrieveData();
public delegate void DeleteData(String sData);
protected void Page_Load(object sender, EventArgs e)
{
//load the initial data..
if (!Page.IsPostBack)
{
if (PopulateDataDelegate != null)
{
this.DataRepeater.DataSource = this.PopulateDataDelegate();
this.DataRepeater.DataBind();
}
}
}
protected void DataRepeater_ItemCommand(object source, RepeaterCommandEventArgs e)
{
if (e.CommandName == "Delete")
{
if (DeleteDataDelegate != null)
{
DeleteDataDelegate((String)e.CommandArgument);
BindDataToRepeater();
}
}
else if (e.CommandName == "Run")
{
String sRunning = (String)e.CommandArgument;
this.ViewDataModalPopupExtender.Hide();
}
}
protected void DataRepeater_ItemDataBound(object source, RepeaterItemEventArgs e)
{
RepeaterItem item = e.Item;
if (item != null && item.DataItem != null)
{
MyDTO oQuery = (MyDTO)item.DataItem;
ImageButton oDeleteControl = (ImageButton) item.FindControl("DeleteData");
ImageButton oRunControl = (ImageButton)item.FindControl("RunData");
if (oDeleteControl != null && oRunControl !=null)
{
oRunControl.ImageUrl = "button_expand.gif";
oRunControl.CommandArgument = "MyID";
if (oQuery !=null)
{
//do something
}
oDeleteControl.ImageUrl = "btn_remove.gif";
oDeleteControl.CommandArgument = "MyID";
}
}
}
public void BindDataToRepeater()
{
this.DataRepeater.DataSource = this.PopulateDataDelegate();
this.DataRepeater.DataBind();
}
public void ShowModal(object sender, EventArgs e)
{
BindDataToRepeater();
this.ViewDataModalPopupExtender.Show();
}
最佳答案
感谢您提醒我为什么我停止使用ASP.NET控件。这正是噩梦的确切类型,它使太多的项目超出了预算和进度。
我的建议是考虑实现此目的的最简单方法。您可以尝试向后弯腰,以使其以ASP.NET的方式工作或采用最短的路线。
您要做的只是生成HTML,它应该从来没有那么困难。
造成问题的最可能原因是ViewState存储在页面中,该页面不会在部分回发时得到更新。因此,随着更新面板中的每一次更改,您都将回发页面的初始 View 状态。
尝试用一个简单的for循环替换中继器(并忽略那些开始提示您不应该将标记和代码混在一起的人)。将数据绑定(bind)语句替换为<%= %>
。
这样一来就消除了 View 状态,应该从重新出现中删除所有删除的行。