Я пытаюсь создать GridView, который будет содержать список пользователей и уровней разрешений, к которым у них есть доступ. Каждая строка будет иметь следующие поля: User_ID, Имя пользователя, Разрешение1, Разрешение2, ..., РазрешениеN, где значения поля разрешений равны «0», если у пользователя нет такого разрешения, и «1», если оно есть (обратите внимание, что столбцы, возвращаемые DAL, не называются Permission1, а являются фактическим именем разрешения). Я хотел бы представить данные с помощью флажков, чтобы администратор мог быстро предоставлять или отзывать разрешения для большого количества пользователей одновременно.
Так как я не буду знать количество разрешений заранее, я динамически создаю TemplateFields и CheckBoxes, и это прекрасно работает; GridView показывает текущие уровни разрешений для всех пользователей. Я сталкиваюсь с проблемой, когда пытаюсь обновить разрешения на основе установки и снятия флажков пользователем.
После того, как пользователь изменил разрешения, у меня есть кнопка «Обновить», которая, конечно же, вызывает обратную передачу. Поскольку обратная передача происходит до события OnClick, к тому времени, когда я достигаю OnClick, все CheckBox'ы были сброшены к своим начальным значениям. Есть ли способ каким-то образом получить значение свойства 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;
}