我有一个在页面加载中动态加载的用户控件:
protected void Page_Load(object sender, EventArgs e)
{
MyControl ctl = (MyControl)LoadControl(controlPath);
ctl.ID = "mycontrol";
this.MyControlPlaceHolder.Controls.Add(ctl);
}
页面的前端:
<asp:PlaceHolder runat="server" ID="MyControlPlaceHolder"></asp:PlaceHolder>
我在页面上有一个启动和回发的点击事件,并调用一个方法,我试图找到控件并设置一些属性:
MyControl ctl = (MyControl)FindControl("mycontrol");
if (ctl != null){
ctl.MyProperty = true;
Response.Write("success");
}
else
Response.Write("fail");
这是在回发后编写失败,所以看起来我在找到控件时做错了。这样做的最佳方式是什么?
编辑:
I switched it to MyControl ctl = (MyControl)this.MyControlPlaceHolder.FindControl("mycontrol");
这使它找到了控件,但是,当回发后控件加载时,看起来好像没有设置属性。
答案 0 :(得分:2)
您必须使用递归FindControl实现,因为FindControl只能查找直接子项。您的控件将插入较低级别的命名容器中。来自MSDN的通用FindControlRecurisve:
private Control FindControlRecursive(Control rootControl, string controlID)
{
if (rootControl.ID == controlID) return rootControl;
foreach (Control controlToSearch in rootControl.Controls)
{
Control controlToReturn =
FindControlRecursive(controlToSearch, controlID);
if (controlToReturn != null) return controlToReturn;
}
return null;
}
或者,如果您的样本中只有一个特定的conatiner:
MyControl ctl = this.MyControlPlaceHolder.FindControl("mycontrol");
if (ctl != null){
ctl.MyProperty = true;
Response.Write("success");
}
else
Response.Write("fail");
ViewState启用您的控件
public class MyControl:Control
{
public bool MyProperty
{
get
{
return ViewState["my"] != null? (bool) ViewState["my"]: false;
}
set
{
ViewState["my"] = value;
}
}
}
答案 1 :(得分:0)
尝试移动代码以将控件动态添加到Init而不是load。我不能确定,但是在Init和Load之间发生了很多事情,如果你的控制不存在并且考虑到它可能会导致这样的问题。
答案 2 :(得分:0)
您在占位符的控件集合中添加了Control。 除了动态创建控件的控制之外,如果你想以这种方式寻找动态添加的控件,你必须从根(可能是页面)开始进行递归搜索,所以,如果你在网上冲浪,你可以找到很好的解决方案。
我个人更喜欢以下解决方案:泛型支持并表示为扩展方法,因此您可以在任何地方使用该解决方案。这些是一些有用的链接