Событие Dropdown_SelectedIndexchange запускается для всех строк в виде сетки
У меня есть gridview, который содержит раскрывающийся список в каждой строке. В раскрывающемся списке выбрано событие изменения индекса. Теперь, когда я изменяю значение в раскрывающемся списке в любой строке сетки, запускается выбранное событие изменения индекса для всех строк в виде сетки. Как решить эту проблему?
Ниже разметка HTML
<asp:GridView ID="gvEditFields" runat="server"
AllowSorting="false" AllowPaging="false" OnRowDataBound="gvEditFields_RowDataBound" >
<Columns>
<asp:TemplateField HeaderText="Project">
<ItemTemplate>
<asp:DropDownList ID="ddlProjectList" runat="server" Width="130px" AutoPostBack="true" OnSelectedIndexChanged="ddlProjectList_SelectedIndexChanged"></asp:DropDownList>
<asp:RequiredFieldValidator ID="rfvProjectList" runat="server" ErrorMessage="* Required"
Display="Dynamic" ControlToValidate="ddlProjectList" SetFocusOnError="True" ValidationGroup="EditGridSave"
Text="Required" ></asp:RequiredFieldValidator>
</ItemTemplate>
</asp:TemplateField>
<asp:TemplateField HeaderText="Task">
<ItemTemplate>
<asp:DropDownList ID="ddlTaskList" runat="server" AppendDataBoundItems="false" Width="150px" OnSelectedIndexChanged="ddlTaskList_SelectedIndexChanged"></asp:DropDownList>
<asp:RequiredFieldValidator ID="rfvTaskList" runat="server" ErrorMessage="* Required"
Display="Dynamic" ControlToValidate="ddlTaskList" SetFocusOnError="True" ValidationGroup="EditGridSave"
Text="Required" ></asp:RequiredFieldValidator>
</ItemTemplate>
</asp:TemplateField>
<asp:TemplateField HeaderText="Task Status">
<ItemTemplate>
<asp:DropDownList ID="ddlTaskStatus" runat="server" Width="100px"></asp:DropDownList>
<asp:RequiredFieldValidator ID="rfvTaskStatus" runat="server" ErrorMessage="* Required"
Display="Dynamic" ControlToValidate="ddlTaskStatus" SetFocusOnError="True" ValidationGroup="EditGridSave"
Text="Required" ></asp:RequiredFieldValidator>
</ItemTemplate>
</asp:TemplateField>
<asp:TemplateField HeaderText="Description">
<ItemTemplate>
<asp:TextBox ID="txtDescription" runat="server" TextMode="MultiLine" Rows="3" Columns="30" />
</ItemTemplate>
</asp:TemplateField>
</Columns>
</asp:GridView>
Это код для привязки базы данных строки и выбранного события изменения индекса.
protected void gvEditFields_RowDataBound(object sender, GridViewRowEventArgs e)
{
if (e.Row.RowType == DataControlRowType.DataRow)
{
SetEditFieldsGridRecords(e.Row);
//DropDownList ddlProjectList = (DropDownList)e.Row.FindControl("ddlProjectList");
}
}
protected void ddlProjectList_SelectedIndexChanged(object sender, EventArgs e)
{
DropDownList ddlProjectList = (DropDownList)sender;
// GridViewRow Row = (GridViewRow)ddlProjectList.NamingContainer;
GridViewRow selectedRow = (GridViewRow)ddlProjectList.Parent.Parent;
int i = selectedRow.RowIndex;
DropDownList ddlTaskList = (DropDownList)selectedRow.Cells[1].FindControl("ddlTaskList");
DropDownList ddlTaskStatus = (DropDownList)selectedRow.Cells[2].FindControl("ddlTaskStatus");
BindTasks(clsCheckDBNull.ToStr(ddlProjectList.SelectedValue), -1, ddlTaskList);
BindStatus(clsCheckDBNull.ToInt(ddlTaskList.SelectedValue), "", ddlTaskStatus);
}
Ниже приведен код используемых методов
private void BindStatus(int intTaskId, string TaskStatus, DropDownList drpStatus)
{
DataTable dt = clsProjectTaskStatuses.SelectAll(BTMSession.AccountID);
DataView dv = new DataView(dt);
drpStatus.DataTextField = "str_PROJECT_TASK_STATUS_NAME";
drpStatus.DataValueField = "int_PROJECT_TASK_STATUS_ID";
drpStatus.DataSource = dv;
drpStatus.DataBind();
SiteUtility.BindTooltip(drpStatus);
if (drpStatus.Items.Count >= 2 && TaskStatus.Trim() != "")
{
if (drpStatus.Items.FindByText(TaskStatus.ToString()) != null)
drpStatus.SelectedIndex = drpStatus.Items.IndexOf(drpStatus.Items.FindByText(TaskStatus.ToString()));
}
}
private void BindTasks(string ProjectNo, int intTaskid, DropDownList drpTasks)
{
if (drpTasks.Items.Count >= 1)
drpTasks.Items.Clear();
DataSet ds = new DataSet();
clsImportFileTypeDAL objImportFileTypeDAL = new clsImportFileTypeDAL();
ds = objImportFileTypeDAL.SelectTaskListByProjectNoUserIDForTimeImport(ProjectNo, clsCheckDBNull.ToInt(BTMSession.UserId));
CommonHelper.DisposeOf(objImportFileTypeDAL);
if (ds != null && ds.Tables.Count > 0 && ds.Tables[0].Rows.Count > 0)
{
drpTasks.DataTextField = "str_TASK_SUMMERY";
drpTasks.DataValueField = "int_task_id";
drpTasks.DataSource = ds;
drpTasks.DataBind();
}
drpTasks.Items.Insert(0, new ListItem("Select Task", "-1"));
SiteUtility.BindTooltip(drpTasks);
drpTasks.SelectedValue = "-1";
if (drpTasks.Items.Count >= 2)
{
if (drpTasks.Items.FindByValue(intTaskid.ToString()) != null)
drpTasks.SelectedIndex = drpTasks.Items.IndexOf(drpTasks.Items.FindByValue(intTaskid.ToString()));
}
}
private void SetEditFieldsGridRecords(строка GridViewRow) { DataRowView drv = (DataRowView)row.DataItem;
DropDownList ddlProjectList = (DropDownList)row.FindControl("ddlProjectList");
BindProjects(clsCheckDBNull.ToStr(drv["str_PROJECT_NO"]), ddlProjectList);
DropDownList ddlTaskList = (DropDownList)row.FindControl("ddlTaskList");
BindTasks(ddlProjectList.SelectedValue, clsCheckDBNull.ToInt(drv["int_TASK_NO"]), ddlTaskList);
DropDownList ddlTaskStatus = (DropDownList)row.FindControl("ddlTaskStatus");
BindStatus(clsCheckDBNull.ToInt(ddlTaskList.SelectedValue), clsCheckDBNull.ToStr(drv["TaskStatus"]), ddlTaskStatus);
BTMWeb.Calender.Calender txtWorkDate = (BTMWeb.Calender.Calender)row.FindControl("txtWorkDate");
DropDownList drpStartHours = (DropDownList)row.FindControl("drpStartHours");
DropDownList drpStartMinutes = (DropDownList)row.FindControl("drpStartMinutes");
DropDownList drpEndHours = (DropDownList)row.FindControl("drpEndHours");
DropDownList drpEndMinutes = (DropDownList)row.FindControl("drpEndMinutes");
TextBox txtDescription = (TextBox)row.FindControl("txtDescription");
int StartHours = clsCheckDBNull.ToDate(drv["StartTime"]).Hour;
int StartMinutes = clsCheckDBNull.ToDate(drv["StartTime"]).Minute;
int EndHours = clsCheckDBNull.ToDate(drv["EndTime"]).Hour;
int EndMinutes = clsCheckDBNull.ToDate(drv["EndTime"]).Minute;
txtWorkDate.DateValue = clsCheckDBNull.ToDate(drv["dt_WORKDATE"]);
drpStartHours.SelectedValue = StartHours.ToString();
drpStartMinutes.SelectedValue = StartMinutes.ToString();
drpEndHours.SelectedValue = EndHours.ToString();
drpEndMinutes.SelectedValue = EndMinutes.ToString();
txtDescription.Text = clsCheckDBNull.ToStr(drv["str_Description"]);
}
Например, если у меня есть строки и если я изменяю значение раскрывающегося списка в первой строке, выбранное изменение индекса срабатывает для обеих строк.
1 ответ
Только одно или два предложения по улучшению:
- Использовать
NamingContainer
свойство, чтобы получить ссылку наGridViewRow
вместоParet.Parent
который по своей природе подвержен ошибкам (например, если вы используете элемент управления вTemplateField
это контейнерный контроль.
Так что вместо:
GridViewRow selectedRow = (GridViewRow)ddlProjectList.Parent.Parent;
этот
GridViewRow selectedRow = (GridViewRow)ddlProjectList.NamingContainer;
- С этим связано: всегда используйте
FindControl
наNamingContainer
контроля, который вы хотите найти, то естьGridViewRow
в этом случае,TableCell
не реализуетINamingContainer
,
Так что вместо:
DropDownList ddlTaskList = (DropDownList)selectedRow.Cells[1].FindControl("ddlTaskList");
этот
DropDownList ddlTaskList = (DropDownList)selectedRow.FindControl("ddlTaskList");
Это просто менее подвержено ошибкам.
Примечание: не понятно, зачем вам нужен такой метод CommonHelper.DisposeOf
, Я полагаю, вы найдете основную проблему в BindTasks
а также BindStatus
, Можешь показать им?