c# - DetailsView 的事件 "ItemUpdating"中的 OldValues 集合始终为空

标签 c# asp.net webforms datasource detailsview

我正在使用 DetailsView,但在更新时,我无法获取 OldValues,因为 ItemUpdating 事件的 DetailsViewUpdateEventArgs.OldValues 始终为空。 NewValues 的值为 ok。

注意:我没有在我的应用程序中使用数据源组件(SqlDataSource、ObjectDataSource、EntityDataSource 等)。在这种情况下,我应该手动做些什么吗?

欢迎任何想法!

谢谢!

补充信息:

我使用的是 ASP.NET 4.0 (WebForms)

我认为以下相关的代码片段:

<asp:DetailsView ID="customerDetails" runat="server" AutoGenerateRows="False" EmptyDataText="No data..." 
    meta:resourcekey="customerDetails" 
    onitemdeleting="customerDetails_ItemDeleting" 
    oniteminserting="customerDetails_ItemInserting" 
    onitemupdating="customerDetails_ItemUpdating" 
    onmodechanging="customerDetails_ModeChanging" 
    CssClass="customerDetails" DataKeyNames="CustomerID">
    <FooterTemplate>
        <asp:LinkButton ID="lnkNew" Text="New" ToolTip="New Customer" CommandName="New" meta:resourcekey="lnkNew" runat="server" />
    </FooterTemplate>
    <EmptyDataTemplate>
        <p><asp:Label ID="lblNoDataHasBeenFound" Text="No data has been found." meta:resourcekey="lblNoDataHasBeenFound" runat="server" /></p>
        <asp:LinkButton ID="lnkNew" Text="New" ToolTip="New Customer" CommandName="New" meta:resourcekey="lnkNew" runat="server" />
    </EmptyDataTemplate>
    <Fields>
        <%--CustomerID--%>
        <asp:TemplateField HeaderStyle-CssClass="detailsHeader" ItemStyle-CssClass="detailsField">
            <HeaderTemplate>
                <asp:Label ID="ltrCustomerIdLabel" meta:resourcekey="ltrCustomerId" Text="CustomerID:" runat="server" />
            </HeaderTemplate>
            <ItemTemplate>
                <asp:Label ID="lblCustomerId" Text='<%# Eval("CustomerID") %>' runat="server"  />
            </ItemTemplate>
            <EditItemTemplate>
                <asp:Label ID="lblCustomerId" Text='<%# Eval("CustomerID") %>' runat="server"  />
            </EditItemTemplate>
            <InsertItemTemplate>
                <asp:TextBox ID="txtCustomerId" Text='<%# Bind("CustomerID") %>' MaxLength="5" width="50px" runat="server" />
                <cc1:DataAnnotationsValidator ID="CustomerIdValidator" Type="String" runat="server" ControlToValidate="txtCustomerId" PropertyName="CustomerID" Text="*" SourceType="DataLayerPOCO.Customer, DataLayerPOCO" CssError="validationError" Display="None"/>
            </InsertItemTemplate>
            <HeaderStyle HorizontalAlign="Right" />
            <ItemStyle HorizontalAlign="Left" />
        </asp:TemplateField>

        <%--CompanyName--%>
        <asp:TemplateField HeaderStyle-CssClass="detailsHeader" ItemStyle-CssClass="detailsField">
            <HeaderTemplate>
                <asp:Literal ID="ltrCompanyNameLabel" meta:resourcekey="ltrCompanyName" Text="Company Name:" runat="server" />
            </HeaderTemplate>
            <ItemTemplate>
                <asp:Literal ID="ltrCompanyName" Text='<%# Eval("CompanyName") %>' runat="server"/>
            </ItemTemplate>
            <EditItemTemplate>
                <asp:TextBox ID="txtCompanyName" Text='<%# Bind("CompanyName") %>' MaxLength="40" Width="310px"
                    runat="server" />
                <cc1:DataAnnotationsValidator ID="CompanyNameValidator" Type="String" runat="server" ControlToValidate="txtCompanyName" PropertyName="CompanyName" Text="*" SourceType="DataLayerPOCO.Customer, DataLayerPOCO" CssError="validationError" Display="None"/>
            </EditItemTemplate>
            <InsertItemTemplate>
                <asp:TextBox ID="txtCompanyName" Text='<%# Bind("CompanyName") %>' MaxLength="40" Width="310px"
                    runat="server" />
                <cc1:DataAnnotationsValidator ID="CompanyNameValidator" Type="String" runat="server" ControlToValidate="txtCompanyName" PropertyName="CompanyName" Text="*" SourceType="DataLayerPOCO.Customer, DataLayerPOCO" CssError="validationError" Display="None"/>
            </InsertItemTemplate>
            <HeaderStyle HorizontalAlign="Right" />

        </asp:TemplateField>
        ...
        ...
        other fields goes here
        ...
        ...
        <%--Edit/Insert--%>
        <asp:TemplateField HeaderStyle-CssClass="detailsHeader" ItemStyle-CssClass="detailsField">
            <ItemTemplate>
                <asp:ImageButton ID="imgEdit" ImageUrl="~/img/pencil.png" meta:resourcekey="imgEdit" AlternateText="Edit" ToolTip="Edit Customer" CommandName="Edit" runat="server" CausesValidation="false"/>
                <asp:ImageButton ID="imgDelete" ImageUrl="~/img/delete.png" meta:resourcekey="imgDelete" AlternateText="Delete" ToolTip="Delete Customer" CommandName="Delete" OnClientClick="return confirmDelete();" runat="server" CausesValidation="false"/>
            </ItemTemplate>
            <EditItemTemplate>
                <asp:ImageButton ID="imgUpdate" ImageUrl="~/img/accept.png" meta:resourcekey="imgUpdate" AlternateText="Update" ToolTip="Update Customer" CommandName="Update" runat="server" />
                <asp:ImageButton ID="imgCancel" ImageUrl="~/img/cancel.png" meta:resourcekey="imgCancel" AlternateText="Cancel" ToolTip="Cancel Edit" CommandName="Cancel" runat="server" CausesValidation="false"/>
            </EditItemTemplate>
            <InsertItemTemplate>
                <asp:ImageButton ID="imgSave" ImageUrl="~/img/disk.png" meta:resourcekey="imgSave" AlternateText="Save" ToolTip="Save Customer Data" CommandName="Insert" runat="server" CausesValidation="true"/>
                <asp:ImageButton ID="imgCancelInsert" ImageUrl="~/img/cancel.png" meta:resourcekey="imgCancelInsert" AlternateText="Cancel" ToolTip="Cancel Insert" CommandName="Cancel" runat="server" CausesValidation="false"/>
            </InsertItemTemplate>
        </asp:TemplateField>
    <Fields>
</asp:DetailsView>

事件 ItemUpdating 如下所示:

    protected void customerDetails_ItemUpdating(object sender, DetailsViewUpdateEventArgs e)
    {
        repository = repository ?? new NorthwindRepositoryEF();

        string id = e.Keys["CustomerID"] as string;

        DataLayerPOCO.Customer customer = repository.GetCustomer(id);

        string companyName = (string)e.NewValues["CompanyName"];
        string contactName = (string)e.NewValues["ContactName"];
        string contactTitle = (string)e.NewValues["ContactTitle"];
        string address = (string)e.NewValues["Address"];
        string city = (string)e.NewValues["City"];
        string region = (string)e.NewValues["Region"];
        string postalCode = (string)e.NewValues["PostalCode"];
        string country = (string)e.NewValues["Country"];
        string phone = (string)e.NewValues["Phone"];
        string fax = (string)e.NewValues["Fax"];

        // Update Customer with the new data
        customer.CompanyName = companyName;
        customer.ContactName = contactName;
        customer.ContactTitle = contactTitle;
        customer.Address = address;
        customer.City = city;
        customer.Region = region;
        customer.PostalCode = postalCode;
        customer.Country = country;
        customer.Phone = phone;
        customer.Fax = fax;

        repository.UpdateCustomer(customer);
        repository.Save();
        //---
        customerDetails.ChangeMode(DetailsViewMode.ReadOnly);
        BindCustomerDetails();
        BindCustomersList();

    }

问题是如果我想处理旧值,我不能:

这个,评估为 0: int oldValuesCount = e.OldValues.Count;

而这个计算结果为 null,即使我将值更改为一个新值: string companyNameOld = e.OldValues["CompanyName"] as string;

最佳答案

似乎只有当您以声明方式绑定(bind)数据源时才会填充旧值。

在这种情况下,您有两种解决方法:

  1. 以编程方式在 ViewState 中保存您可能需要的任何旧值,并在需要时检索它。这节省了对数据库的访问。

  2. 更好的选择是使用您在 OnItemUpdating 中定义的客户变量,因为它从数据库查询信息并包含所有旧值。

希望对您有所帮助。

关于c# - DetailsView 的事件 "ItemUpdating"中的 OldValues 集合始终为空,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6177876/

相关文章:

testing - 如何测试网络表单?

c# - 地址之间的距离

c# - 为什么 LINQ Where 搜索查询在 List<> 上比 ConcurrentBag<> 更快

c# - 可空类型是否适合我的情况?

c# - 我应该使用什么类型的变量?

c# - 如何遍历用户控件中的控件找到某个控件?

asp.net - 将 Angular2 应用程序包含到 Asp.Net Webforms 页面

c# - 使用c#从图像中提取子图像

c# - 使用其关系中的属性为子类建模的正确方法

c# - 路由声明它在 system.web 中定义但未找到错误