GridView RowDataBound Handler - Не удается получить данные из строки - PullRequest
3 голосов
/ 19 июля 2011

У меня есть GridView с анонимным типом. Мне нужно проверить значение в ячейке и выделить эту ячейку, если выполняется условие. Проблема в том, что я всегда получаю пустую строку, когда пытаюсь извлечь данные из ячеек строки. Я успешно выделил все ячейки и проверил в отладчике Visual Studio 2010 и подтвердил, что в строке есть данные (в DataItem строки есть нужные мне значения). Это происходит на PostBack, я не уверен, является ли это проблемой или нет.

Вот код и решения, которые я пробовал:

protected void grvValidCourses_RowDataBound(Object sender, GridViewRowEventArgs e) {
 if (e.Row.RowType == DataControlRowType.DataRow) {
  String str = e.Row.Cells[6].Text.ToString(); // empty string
  Label lbl = (Label) grvValidCourses.FindControl("lblEndDate"); // null
  DataRowView rowView = (DataRowView)e.Row.DataItem; // exception about casting anonymous type

Что здесь происходит? Почему я не могу получить данные из ячеек?

Разметка для GridView:

 <asp:GridView ID="grvValidCourses" runat="server" Width="790px" OnRowCancelingEdit="grvValidCourses_RowCancelingEdit"
                    OnRowEditing="grvValidCourses_RowEditing" OnRowUpdating="grvValidCourses_RowUpdating" OnRowDeleting="grvValidCourses_RowDeleting"
                    AutoGenerateColumns="False" OnSelectedIndexChanged="grvValidCourses_SelectedIndexChanged"
                    OnRowDataBound="grvValidCourses_RowDataBound" >
                    <Columns>
                        <asp:CommandField ShowEditButton="True" EditText="Edit" UpdateText="Update |" />
                        <asp:TemplateField ShowHeader="False">
                             <ItemTemplate>
                               <asp:LinkButton ID="lbnDelete" runat="server" CausesValidation="False" CommandName="Delete"
                                    Text='<%# (Eval("active") == null ? "Delete" : ((Eval("active").ToString() == "0" ? "Restore" : "Delete"))) %>' />  
                             </ItemTemplate>
                        </asp:TemplateField>
                        <asp:CommandField ShowSelectButton="True" SelectText="Details" />
                        <asp:TemplateField HeaderText="Training Name" SortExpression="coursename">
                            <EditItemTemplate>
                                <asp:Label ID="Label1" runat="server" Text='<%# Eval("coursename") %>'></asp:Label>
                            </EditItemTemplate>
                            <ItemTemplate>
                                <asp:Label ID="Label2" runat="server" Text='<%# Bind("coursename") %>'></asp:Label>
                            </ItemTemplate>
                        </asp:TemplateField>
                        <asp:BoundField DataField="ttNo" HeaderText="#" SortExpression="ttNo" ReadOnly="True" />
                        <asp:TemplateField HeaderText="Course Date" SortExpression="startDate">
                            <EditItemTemplate>
                                <asp:TextBox ID="TextBox3" runat="server" Text='<%# Bind("startdate", "{0:M/d/yyyy}") %>'></asp:TextBox>
                                <asp:CalendarExtender ID="TextBox3_CalendarExtender" runat="server" Enabled="True"
                                    TargetControlID="TextBox3" Format="M/d/yyyy">
                                </asp:CalendarExtender>
                            </EditItemTemplate>
                            <ItemTemplate>
                                <asp:Label ID="Label3" runat="server" Text='<%# Bind("startdate", "{0:M/d/yyyy}") %>'></asp:Label>
                            </ItemTemplate>
                        </asp:TemplateField>
                        <asp:TemplateField HeaderText="Expiry Date" SortExpression="endDate">
                            <EditItemTemplate>
                                <asp:TextBox ID="TextBox1" runat="server" Text='<%# Bind("enddate", "{0:M/d/yyy}") %>'></asp:TextBox>
                                <asp:CalendarExtender ID="TextBox1_CalendarExtender" runat="server" Enabled="True"
                                    TargetControlID="TextBox1" Format="M/d/yyyy">
                                </asp:CalendarExtender>
                            </EditItemTemplate>
                            <ItemTemplate>
                                <asp:Label ID="lblEndDate" runat="server" Text='<%# Bind("enddate", "{0:M/d/yyyy}") %>'></asp:Label>
                            </ItemTemplate>
                        </asp:TemplateField>
                    </Columns>
                    <EmptyDataTemplate>No valid courses found.</EmptyDataTemplate>
                </asp:GridView>

ОБНОВЛЕНИЕ: я пробовал все ваши предложения, но я получаю исключения или нули или пустые строки. Я даже воссоздал проблему в более простом примере и до сих пор не могу понять это! Я буду продолжать пытаться, и я ценю любые новые предложения.

Ответы [ 4 ]

16 голосов
/ 19 июля 2011

Часть 1 - Отсутствует текст

Возможно, в первой части вы получаете пустые значения из-за необходимости доступа к дочернему элементу управления:

String str = ((DataBoundLiteralControl)e.Row.Cells[6].Controls[0]).Text;

Чтобы проверить, есть ли в ваших ячейках какие-либозначения в режиме отладки (проверьте текст в окне вывода отладки):

void grvValidCourses_RowDataBound(object sender, GridViewRowEventArgs e)
{
  if (e.Row.RowType == DataControlRowType.DataRow)
  {
    System.Diagnostics.Trace.WriteLine(e.Row.Cells.Count);
     foreach (TableCell c in e.Row.Cells)
     {
      System.Diagnostics.Trace.WriteLine(c.Text);
     }
  }
}

Часть 2. Отсутствует управление

Это неправильно:

Label lbl = (Label) grvValidCourses.FindControl("lblEndDate"); // null

Вы не можетеискать в сетке для контроля строки.Вам нужно выполнить поиск в строке.

Label lblProductOptionGrpName = (Label)e.Row.FindControl("lblProductOptionGrpName");

Часть 3 - Доступ к DataItem

DataBinder.Eval(e.Row.DataItem, "ColumnName")

Наконец, я не уверен, что вы делаете с вашими анонимными типами,но вам может потребоваться проверить содержимое перед доступом к свойствам:

if(MyControl.GetType() == typeof(HyperLink))
{
  HyperLink TestLink = (HyperLink)MyControl;
  TestLink .Visible = false;
}
3 голосов
/ 19 июля 2011

Хорошо, я наконец понял, что происходит!Я могу получить доступ только к связанному полю через строку (например, e.Row.Cells[index].Text).Поэтому я использовал связанное поле, чтобы получить идентификатор своего элемента, затем нашел элемент в базе данных, получил дату, сравнил ее и выделил ячейки строки.Не самый эффективный способ, но он работает.

Код из простого примера:

Внешний вид таблицы вида

<asp:GridView ID="gdv" runat="server" AutoGenerateColumns="True" OnSelectedIndexChanged="gdv_SelectedIndexChanged">
        <Columns>
            <asp:CommandField ShowSelectButton="True" SelectText="Select" />
            <asp:BoundField DataField="id" HeaderText="#" ReadOnly="True" />
            <asp:TemplateField HeaderText="CategoryName" >
                <ItemTemplate>
                    <asp:Label ID="Label2" runat="server" Text='<%# Bind("name") %>'></asp:Label>
                </ItemTemplate>
            </asp:TemplateField>
            <asp:TemplateField HeaderText="Description" >
                <ItemTemplate>
                    <asp:Label ID="Label3" runat="server" Text='<%# Bind("desc") %>'></asp:Label>
                </ItemTemplate>
            </asp:TemplateField>
        </Columns>
    </asp:GridView>
    <asp:Label ID="lblSelected" runat="server" Text="No row selected"></asp:Label>

КодЗа страницей

protected void Page_Load(object sender, EventArgs e)
{
    if (!IsPostBack) {
        var qry = ctx.Categories
              .Select(c => new {
                  id = c.CategoryID,
                  name = c.CategoryName,
                  desc = c.Description,
              });
        gdv.DataSource = qry;
        gdv.DataBind();
    }
}

protected void gdv_SelectedIndexChanged(object sender, EventArgs e) {
    selectRow();
}

private void selectRow() {
    GridViewRow row = gdv.SelectedRow;
    String strId = row.Cells[1].Text; // Bound Field column
    lblSelected.Text = strId;
    // use ID to get object from database...
}
2 голосов
/ 19 июля 2011

Если вы можете опубликовать разметку, то можно будет выяснить вашу проблему.

Вы делаете это неправильно,

это DataRowView rowView = (DataRowView)e.Row.DataItem;

должно быть DataRow rowView = ((DataRowView)e.Row.DataItem).Row;

0 голосов
/ 10 апреля 2017
protected void gridPanne_RowDataBound(object sender, GridViewRowEventArgs e)
{
  if(e.Row.RowType==DataControlRowType.DataRow)
    {
        Label label = (Label)e.Row.FindControl("Label4");
        lbl.Text += "  ** / **  "+label.Text;
    }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...