在动态 GridView TemplateField 中保留复选框的值

发布于 2024-12-02 17:12:19 字数 3247 浏览 4 评论 0原文

我正在尝试创建一个 GridView,其中包含用户及其有权访问的权限级别的列表。每行将包含以下字段:User_ID、用户名、Permission1、Permission2、...、PermissionN,其中,如果用户没有该权限,则权限字段的值为“0”;如果用户有该权限,则权限字段的值为“1”(请注意, DAL 返回的列未命名为 Permission1,而是权限的实际名称)。我想使用复选框来表示数据,以允许管理员快速向大量用户授予或撤销权限。

由于我事先不知道权限的数量,因此我动态创建 TemplateFields 和 CheckBoxes,这工作得很好; GridView 显示所有用户当前的权限级别。当我尝试根据用户选中和取消选中的框更新权限时遇到问题。

一旦用户完成权限更改,我就会有一个“更新”按钮,这当然会导致回发。由于回发发生在 OnClick 事件之前,因此当我到达 OnClick 时,所有复选框都已重置为其初始值。有没有办法在回发发生之前以某种方式获取 CheckBoxes 的 Checked 属性的值?还有另一种(更好的)方法吗?谢谢。

ASPX:

<asp:GridView ID="gvUsers" runat="server" AutoGenerateColumns="False"
EnableModelValidation="True" DataKeyNames="ID">
</asp:GridView>
<asp:ObjectDataSource ID="odsUsers" runat="server" SelectMethod="GET_USERS"
TypeName="oDAL">
</asp:ObjectDataSource>
<asp:Button ID="btnUpdate" runat="server" Text="Update Permissions" OnClick="btnUpdate_OnClick"/>

代码隐藏:

private static string[] excludeCols = { "ID", "Username" };
protected void Page_Load(object sender, EventArgs e)
{
    if (!IsPostBack) // Removing the IsPostBack check refreshes the GridView when the Update button is clicked
        bindGridViewWithHiddenID(gvUsers, ((DataView)odsUsers.Select()).Table, excludeCols);
}

public static bool bindGridViewWithHiddenID(GridView gv, DataTable dt, string[] excludeCols)
{
    gv.Columns.Clear();
    gv.DataBind();
    if (gv != null && dt != null && dt.Rows.Count > 0)
    {
        DataControlField newField;
        foreach (DataColumn column in dt.Columns)
        {
            if (excludeCols.Contains(column.ColumnName))
            {
                newField = new BoundField();
                ((BoundField)newField).DataField = column.ColumnName;
            }
            else
            {
                newField = new TemplateField();
                ((TemplateField)newField).ItemTemplate = new CustomTemplate(column.ColumnName);
            }
            newField.HeaderText = column.ColumnName;
            if (column.ColumnName == "ID" || column.ColumnName.EndsWith("_ID"))
                newField.Visible = false;
            gv.Columns.Add(newField);
        }

        gv.DataSource = dt;
        gv.DataBind();
        return true;
    }
    return false;

}

// By this time execution reaches here the CheckBoxes have already been reset
protected void btnUpdate_Click(object sender, EventArgs e)
{
    ...
}

CustomTemplate 类:

public class CustomTemplate : ITemplate
{
private string binding;
private static int count = 0;
public CustomTemplate(string colNameBinding)
{
    binding = colNameBinding;
}

public void InstantiateIn(Control container)
{
    CheckBox chk = new CheckBox();
    chk.ID = "chk" + count++;
    chk.DataBinding += new EventHandler(this.chk_OnDataBinding);
    container.Controls.Add(chk);
}

public void chk_OnDataBinding(object sender, EventArgs e)
{
    CheckBox chk = (CheckBox)sender;
    GridViewRow namingContainer = (GridViewRow)chk.NamingContainer;
    DataRowView dataRow = (DataRowView)namingContainer.DataItem;

    if (dataRow[binding].ToString() == "1")
        chk.Checked = true;
    else
        chk.Checked = false;
}

I am trying to create a GridView which will contain a list of users and permission levels that they have access to. Each row will have the following fields: User_ID, Username, Permission1, Permission2, ..., PermissionN, where the values of permissions field are "0" if the user does not have that permission and "1" if they do (note that columns returned by the DAL are not named Permission1, but rather are the actual name of the permission). I would like to represent the data by using CheckBoxes, to allow an admin to quickly grant or revoke permissions to a large number of users at once.

Since I will not know the number of permissions before-hand, I dynamically create TemplateFields and CheckBoxes, and this works fine; the GridView shows the current permission levels of all the users. I run into a problem when I try to update permissions based on the user checking and unchecking boxes.

Once the user is done changing permissions, I have an "Update" button, which of course causes a postback. Since the postback occurs before the OnClick event, by the time I reach OnClick all of the CheckBoxes have been reset to their initial values. Is there a way I can somehow grab the value of the CheckBoxes' Checked property before the postback occurs? Is there another (better) way of doing this? Thanks.

ASPX:

<asp:GridView ID="gvUsers" runat="server" AutoGenerateColumns="False"
EnableModelValidation="True" DataKeyNames="ID">
</asp:GridView>
<asp:ObjectDataSource ID="odsUsers" runat="server" SelectMethod="GET_USERS"
TypeName="oDAL">
</asp:ObjectDataSource>
<asp:Button ID="btnUpdate" runat="server" Text="Update Permissions" OnClick="btnUpdate_OnClick"/>

Code Behind:

private static string[] excludeCols = { "ID", "Username" };
protected void Page_Load(object sender, EventArgs e)
{
    if (!IsPostBack) // Removing the IsPostBack check refreshes the GridView when the Update button is clicked
        bindGridViewWithHiddenID(gvUsers, ((DataView)odsUsers.Select()).Table, excludeCols);
}

public static bool bindGridViewWithHiddenID(GridView gv, DataTable dt, string[] excludeCols)
{
    gv.Columns.Clear();
    gv.DataBind();
    if (gv != null && dt != null && dt.Rows.Count > 0)
    {
        DataControlField newField;
        foreach (DataColumn column in dt.Columns)
        {
            if (excludeCols.Contains(column.ColumnName))
            {
                newField = new BoundField();
                ((BoundField)newField).DataField = column.ColumnName;
            }
            else
            {
                newField = new TemplateField();
                ((TemplateField)newField).ItemTemplate = new CustomTemplate(column.ColumnName);
            }
            newField.HeaderText = column.ColumnName;
            if (column.ColumnName == "ID" || column.ColumnName.EndsWith("_ID"))
                newField.Visible = false;
            gv.Columns.Add(newField);
        }

        gv.DataSource = dt;
        gv.DataBind();
        return true;
    }
    return false;

}

// By this time execution reaches here the CheckBoxes have already been reset
protected void btnUpdate_Click(object sender, EventArgs e)
{
    ...
}

CustomTemplate class:

public class CustomTemplate : ITemplate
{
private string binding;
private static int count = 0;
public CustomTemplate(string colNameBinding)
{
    binding = colNameBinding;
}

public void InstantiateIn(Control container)
{
    CheckBox chk = new CheckBox();
    chk.ID = "chk" + count++;
    chk.DataBinding += new EventHandler(this.chk_OnDataBinding);
    container.Controls.Add(chk);
}

public void chk_OnDataBinding(object sender, EventArgs e)
{
    CheckBox chk = (CheckBox)sender;
    GridViewRow namingContainer = (GridViewRow)chk.NamingContainer;
    DataRowView dataRow = (DataRowView)namingContainer.DataItem;

    if (dataRow[binding].ToString() == "1")
        chk.Checked = true;
    else
        chk.Checked = false;
}

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

生生不灭 2024-12-09 17:12:19

这种行为是预期的并且是正常的。它与页面生命周期和动态添加控件上的 ViewState 跟踪有关。我建议您阅读 Scott Mitchell 的这篇文章,特别是标题为:查看状态和动态添加的控件。

This behavior is expected and is normal. It has to do with the Page Life cycle and tracking of ViewState on dynamically added controls. I recommend you read this article from Scott Mitchell, specially the section titled: View State and Dynamically Added Controls.

无需解释 2024-12-09 17:12:19

我尝试将 GridView 绑定添加到 OnInit() 中,但由于在 GridView 进行 DataBound 之前,CheckBox 并未实际实例化,并且由于 DataBinding 会将 GridView 恢复到其原始状态,因此我似乎再次陷入困境。

I tried adding the GridView binding to OnInit() but since the CheckBoxes aren't actually instantiated until the GridView is DataBound, and since DataBinding will revert the GridView back to its original state, I seem to be stuck again.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文