我有一个列出当前项目的页面,如果您选择其中一个项目,它会使用参数加载相同的页面,并显示属于该项目的用户。出于某种原因,它有时(取决于项目)在文本框中显示用户名应该是项目的名称。
protected void Page_Load(object sender, EventArgs e)
{
var dbProxy = new DatabaseProxy();
// No parameter, show project list
if (this.Request.QueryString["ProjectID"] == null)
{
var user = new UserContainer();
user.LoadUser(HttpContext.Current.User.Identity.Name);
List<ProjectContainer> projects = dbProxy.GetProjects(user.UserID, true);
foreach (ProjectContainer project in projects)
{
var linkButton = new Button {Text = project.ProjectName};
this.ProjectsPanel.Controls.Add(linkButton);
}
}
// We have a parameter, show users for project
else
{
int projectId = int.Parse(this.Request.QueryString["ProjectID"]);
this.LoadUsers(projectId);
}
}
LoadUsers功能:
private void LoadUsers(int projectId)
{
var databaseProxy = new DatabaseProxy();
List<UserContainer> users = databaseProxy.GetUsersByProject(projectId);
foreach (UserContainer user in users)
{
var txtUsername = new TextBox();
txtUsername.Text = user.Username; // <- Shows projectname!
var txtUsername2 = new TextBox();
txtUsername2.Text = user.Username; // <- Shows username!
this.UserPanel.Controls.Add(txtUsername);
this.UserPanel.Controls.Add(txtUsername2);
}
}
HTML代码:
<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="EditUsers.aspx.cs" Inherits="xxx.EditUsers1" MasterPageFile="PageStructure.Master" %>
<asp:Content ID="Content1" runat="server" ContentPlaceHolderID="ContentPlace">
<asp:Panel ID="Panel1" runat="server" style="padding-left: 25px">
<div style="margin-bottom: 50px">
<asp:Label ID="lblHeader" runat="server" Text="Benutzer verwalten" CssClass="PageHeader"></asp:Label>
</div>
<asp:Panel ID="UserPanel" runat="server"></asp:Panel>
<asp:Panel ID="ProjectsPanel" runat="server"></asp:Panel>
</asp:Panel>
</asp:Content>
母版页只包含页眉和页脚以及一些不同页面的JavaScript。
输出:
http://i.imgur.com/wAxTSjT.png
用户变量包含正确的值,LoadUsers之后不再发生任何事情。我甚至可以添加一个空标签作为第一个元素,然后它将在第一个文本框中显示正确的用户名!我不知道为什么,但它看起来真的想要我添加的第一个元素中的项目名称。
示例如何运作:
private void LoadUsers(int projectId)
{
var databaseProxy = new DatabaseProxy();
List<UserContainer> users = databaseProxy.GetUsersByProject(projectId);
foreach (UserContainer user in users)
{
var lbl = new Label();
lbl.Text = ""; <- Stays invisible!
var txtUsername = new TextBox();
txtUsername.Text = user.Username; // <- Shows username!
var txtUsername2 = new TextBox();
txtUsername2.Text = user.Username; // <- Shows username!
this.UserPanel.Controls.Add(lbl);
this.UserPanel.Controls.Add(txtUsername);
this.UserPanel.Controls.Add(txtUsername2);
}
}
我想我可以找到一种解决方法,比如将它拆分为两种不同的Web表单,但我仍然想知道为什么我得到错误的值。
答案 0 :(得分:2)
问题是由控件的不同步视图状态引起的。
在ASP.NET中,每个表单控件都有一个视图状态,用于跟踪输入的值,以便在回发后重新加载页面时,它可以自动将以前的值填充到每个表单控件中。通常这可以正常工作,通常是理想的行为。但是,在某些情况下(比如这一个)动态生成控件并且在从post-back返回后它们的值发生了变化,那么它们的视图状态最终会覆盖这些值,从而导致出现问题。
要解决此问题,请为动态生成的控件禁用回发,如下所示:
foreach (UserContainer user in users)
{
var txtUsername = new TextBox();
txtUsername.EnableViewState = false;
txtUsername.Text = user.Username; // <- Shows projectname!
var txtUsername2 = new TextBox();
txtUsername2.EnableViewState = false;
txtUsername2.Text = user.Username; // <- Shows username!
this.UserPanel.Controls.Add(txtUsername);
this.UserPanel.Controls.Add(txtUsername2);
}
或者每次生成每个控件时为其分配一个不同的ID,以便控件永远不会与viewstate中保存的数据的ID匹配,如下所示:
foreach (UserContainer user in users)
{
var txtUsername = new TextBox();
txtUsername.ID = Guid.NewGuid().ToString("N");
txtUsername.Text = user.Username; // <- Shows projectname!
var txtUsername2 = new TextBox();
txtUsername2.ID = Guid.NewGuid().ToString("N");
txtUsername2.EnableViewState = false;
txtUsername2.Text = user.Username; // <- Shows username!
this.UserPanel.Controls.Add(txtUsername);
this.UserPanel.Controls.Add(txtUsername2);
}
警告:如果您使用上述任一方法,则在回发或表单提交期间不会发回控件的值。因此,任何这些控件中任何用户输入的值都将丢失。要避免此问题,请使用Javascript将这些值复制到表单提交上的隐藏控件(即不动态生成)。或者只是简单地不使用动态生成的控件(如果可能的话)。
有关视图状态与动态控件的详细信息,请查看:http://msdn.microsoft.com/en-us/library/hbdfdyh7%28v=vs.100%29.aspx