DataGrid动态生成模板列,模板列里面有一个TextBox,如何获取用户在TextBox里面输入的值啊?

zhangmeiping 2005-03-18 05:16:24
这个问题已经有人问过,但是没有看到真正解决的方法。

Datagrid中动态生成的列,ASP.NET不为其维持状态,也就是在Postback后会自动消失,只要页面一提交所有动态生成的列就消失了。

所以要在Page_Load中自己重新动态生成列,而且不管IsPostBack属性的值都生成。

这样就会有个问题,当点击确定按钮的时候,页面提交,调用Page_Load()重新动态生成Datagrid的列,那用户在TextBox里输入的值就消失了。

如果是显示定义的DataGrid的模板列就没问题,不用在Page_Load里重新生成列,用户在TextBox输入的值也会保存下来。

相关帖子:
http://community.csdn.net/Expert/topic/3607/3607808.xml?temp=.2782099
http://community.csdn.net/Expert/topic/3466/3466563.xml?temp=.8329126
...全文
484 24 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
24 条回复
切换为时间正序
请发表友善的回复…
发表回复
hchxxzx 2005-06-22
  • 打赏
  • 举报
回复
思归的“动态控件的状态问题”

http://blog.joycode.com/saucer/archive/2004/10/20/35927.aspx



希望写下这样的logic对大家做troubleshooting有所帮助。我们可以这样分析这个问题:



第一步,简化一下Page。建一个新的ASP.Net Web Application,用下面的code:



private void Page_Load(object sender, System.EventArgs e)

{

DropDownList ddlDynamic = new DropDownList();

ddlDynamic.ID = "ddlDynamic";

HtmlForm form1 = (HtmlForm)this.FindControl("Form1");

if (!IsPostBack)

{

ddlDynamic.Items.Add("Before");

}

form1.Controls.Add(ddlDynamic);

if (!IsPostBack)

{

ddlDynamic.Items.Add("After");

}

}



在Page上扔个Button,以便可以PostBack。运行后Postback的结果,“Before” item没被保留,“After”被保留了。问题被isolate:问题不在于DropDownList或者ListCollection对于view state的处理出问题,而是特定一个ListItem view state的处理有异。



现在有目标了,接下来看ListItem source code:



internal object SaveViewState()

{

if (this.misc.Get(2) && this.misc.Get(3))

{

return new Pair(this.Text, this.Value);

}

if (this.misc.Get(2))

{

return this.Text;

}

if (this.misc.Get(3))

{

return new Pair(null, this.Value);

}

return null;

}



可以看到只有misc.Get(2)或misc.Get(3)符合一定条件才存view state,鉴于misc是private member,继续在ListItem的code里找什么会影响misc.Get(2) or misc.Get(3)的值,结果如下:



internal bool Dirty

{

set

{

this.misc.Set(2, value);

this.misc.Set(3, value);

}

}



找到了唯一的可能,在Reflector里看Set方法的Callee Graph,找到System.Web.UI.WebControls.ListItemCollection.Add(ListItem):Void方法。接续看source code:



public void Add(ListItem item)

{

this.listItems.Add(item);

if (this.marked)

{

item.Dirty = true;

}

}



这里有一个private bool marked flag。继续在ListItemCollection里找:



internal void TrackViewState()

{

this.marked = true;

for (int num1 = 0; num1 < this.Count; num1++)

{

this[num1].TrackViewState();

}

}



void IStateManager.TrackViewState()

{

this.TrackViewState();

}



好了,看来这个方法就是关键了……由于是Interface的方法,我们可以尝试一下在Page2里调用它:



((IStateManager)(ddlDynamic.Items)).TrackViewState();

if (!IsPostBack)
{
for (int i=1; i <=3; i++)
ddlDynamic.Items.Add(new ListItem(i.ToString(), i.ToString()));
}
form1.Controls.Add(ddlDynamic);



确实是起作用了……鉴于如此,我们可以猜测ControlCollection.Add一定调用了System.Web.UI.WebControls.ListItemCollection.TrackViewState()这个方法。要证明这点容易多了……



用WinDbg,在System.Web.UI.WebControls.ListItemCollection.TrackViewState()方法上设个断点。Call Stack如下:



019cf8b0 06538fd0 [DEFAULT] [hasThis] Void System.Web.UI.WebControls.ListItemCollection.TrackViewState()
019cf8b4 06538fbe [DEFAULT] [hasThis] Void System.Web.UI.WebControls.ListControl.TrackViewState()
019cf8bc 06538e53 [DEFAULT] [hasThis] Void System.Web.UI.Control.InitRecursive(Class System.Web.UI.Control)
019cf8d8 0653758a [DEFAULT] [hasThis] Void System.Web.UI.Control.AddedControl(Class System.Web.UI.Control,I4)
019cf8f4 06537462 [DEFAULT] [hasThis] Void System.Web.UI.ControlCollection.Add(Class System.Web.UI.Control)
019cf904 063c06fc [DEFAULT] [hasThis] Void WebApplication37.WebForm3.Page_Load(Object,Class System.EventArgs)
at [+0x13c] [+0x8c] c:\inetpub\wwwroot\webapplication37\webform3.aspx.cs:36
019cf944 065391a4 [DEFAULT] [hasThis] Void System.Web.UI.Control.OnLoad(Class System.EventArgs)





好了……一切都明了……
liuqinglq 2005-06-22
  • 打赏
  • 举报
回复
这样的问题……

楼主不知用过ASP没?

ASP里可没有什么viewstate,那要保存用户之前输入的值该怎么办?

这里也是一样的

用Request.From[]就可以了
sunnystar365 2005-06-22
  • 打赏
  • 举报
回复
MARK
loepin 2005-06-22
  • 打赏
  • 举报
回复
怎么解决的?
renyu732 2005-06-22
  • 打赏
  • 举报
回复
好东西.
yuzuhe 2005-03-25
  • 打赏
  • 举报
回复
gotdotnet.com
petertangh 2005-03-25
  • 打赏
  • 举报
回复
我又有新问题了。这样做之后,排序点了datagrid就不见了。
老大再帮帮忙呀。
petertangh 2005-03-25
  • 打赏
  • 举报
回复
呵呵,问题解决了。
五星的就是不一样。
xrll 2005-03-25
  • 打赏
  • 举报
回复
http://218.84.107.5/antai/g.aspx
点编辑\更新看看!
LoveCherry 2005-03-25
  • 打赏
  • 举报
回复
http://blog.csdn.net/xdev/archive/2004/12/29/233146.aspx
你是在运行时创建控件, 那么, 在PostBack的时候, ASP.NET是无法取得这些值的. 为此, 我们只能通过Request.Form的方法来得到这些值
petertangh 2005-03-25
  • 打赏
  • 举报
回复
我也遇到同样的问题。
难道动态生成的就只能显示吗?那还用模板干嘛。
不如直接new boundcolumn算了。
应该有办法解决吧。高手都跑到哪里去了。
lovelxj 2005-03-24
  • 打赏
  • 举报
回复
把这些值利用隐含的textBOX来存放
或者用页面的CACH 来存放

动态生成的东西状态是可能是不能保存的,除非你把他放在CACH中去
jacklin_036 2005-03-24
  • 打赏
  • 举报
回复
down
fds2003 2005-03-24
  • 打赏
  • 举报
回复
UP
yc_zhj 2005-03-24
  • 打赏
  • 举报
回复
我跟你遇到了相同的问题,正在想怎么解决了,如果仁兄有方法了,告诉一声撒。yc-zhj@21cn.com
HONGDUN 2005-03-24
  • 打赏
  • 举报
回复
up
zhangmeiping 2005-03-21
  • 打赏
  • 举报
回复
EditTemplateColumn是自定义的模板列:

public class EditTemplateColumn: ITemplate
{
public EditTemplateColumn(string colname)
{
m_strColName=colname;
}

public void InstantiateIn(Control container)
{
TextBox edt = new TextBox();
edt.DataBinding += new EventHandler(this.OnDataBinding);

container.Controls.Add(edt);
}

public void OnDataBinding(object sender, EventArgs e)
{
TextBox edt = (TextBox) sender;

DataGridItem container = (DataGridItem) edt.NamingContainer;

edt.Text = ( (DataRowView)container.DataItem )[m_strColName].ToString();
}


private string m_strColName;
}
fanruinet 2005-03-21
  • 打赏
  • 举报
回复
EditTemplateColumn是从哪里来的?
zhangmeiping 2005-03-21
  • 打赏
  • 举报
回复
先贴出我的:

private void Page_Load(object sender, System.EventArgs e)
{

if( !this.IsPostBack )
{
...
}

TemplateColumn col=new TemplateColumn();
col.ItemTemplate=new EditTemplateColumn("field_name");
col.HeaderText="name";

datagrid1.Columns.Add(col);

datagrid1.DataSource=ds;
datagrid1.DataBind();

}

在按钮提交里面测试columns属性:
private void btnSubmit_Click(object sender, System.EventArgs e)
{
int n=datagrid1.Columns.Count;
}
n的数目总是aspx里面声明的列的数目,而不包含动态生成的列。


在Page_load里,不判断IsPostBack属性,一提交就动态生成模板列,这样用户输入的数据自然随着重新生成模板列而消失。

如果把在if( !this.IsPostBack ) 里面动态生成模板列,如下:
if( !this.IsPostBack )
{
TemplateColumn col=new TemplateColumn();
col.ItemTemplate=new EditTemplateColumn("field_name");
col.HeaderText="name";

datagrid1.Columns.Add(col);

datagrid1.DataSource=ds;
datagrid1.DataBind();
}
测试datagrid1.Columns.Count的数目还是一样的。因为一点击按钮,列已经消失了。
zhangmeiping 2005-03-21
  • 打赏
  • 举报
回复
我说的动态生成的列,是指不是在aspx文件里面直接声明的列,如下:
<Columns>
<asp:BoundColumn DataField="costcodename_1" HeaderText="aa"></asp:BoundColumn>
<asp:BoundColumn DataField="costcodename" HeaderText="bb"></asp:BoundColumn>
</Columns>

而是在程序里用代码的方法生成的列:
TemplateColumn col=new TemplateColumn();
col.ItemTemplate=new EditTemplateColumn("field_name");
col.HeaderText="name";

datagrid1.Columns.Add(col);

EditTemplateColumn是自定义的模板列,里面自动生成一个textbox控件。

用程序生成的列,只要页面一提交就会消失。也就是说,即使点击一个没有编写单击事件代码的按钮,页面一

提交,生成列的就消失了。 这种情况可以设断点测试datagrid1.Columns集合就可以看到。

集合里面只有在aspx里面声明的列,而没有代码里生成的列。

如果把生成列的代码放在 !Page.IsPostBack 里面,点击按钮提交都不会执行生成列的代码,

datagrid1.Columns更加不会有动态生成的列。

如果把生成列的代码放在Page.IsPostBack里面,点击按钮提交以后虽然生成了动态的列,但是用户输入的数据

因为原来的列已经消失,生成了新的textbox, 输入的数据也自然消失了。

如果谁有已经测试成功的代码请贴上来谢谢。
加载更多回复(4)

62,244

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术交流专区
javascript云原生 企业社区
社区管理员
  • ASP.NET
  • .Net开发者社区
  • R小R
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

.NET 社区是一个围绕开源 .NET 的开放、热情、创新、包容的技术社区。社区致力于为广大 .NET 爱好者提供一个良好的知识共享、协同互助的 .NET 技术交流环境。我们尊重不同意见,支持健康理性的辩论和互动,反对歧视和攻击。

希望和大家一起共同营造一个活跃、友好的社区氛围。

试试用AI创作助手写篇文章吧