如何在 HttpPost 验证错误后重新冻结 ViewModel



在 HttpPost 上失败后,我的视图模型未正确解除冻结,导致在重新显示视图时在"购物车项目"上出现"对象引用未设置为对象的实例"错误。

如何冻结我的 ViewModel,以便它可以重新显示错误消息?

我尝试使用 @Html.HiddenFor(...) 来保留未显示的值(根据此答案),但这不起作用。

视图模型

    public class CheckoutViewModel
    {
        public List<Cart> CartItems { get; set; }
        [DisplayFormat(DataFormatString = "${0:F2}")]
        public double CartTotal { get; set; }
        public virtual Order Order { get; set; }
    }

控制器 GET:操作结果

    //GET: /Checkout/AddressAndPayment
    public ActionResult AddressAndPayment()
    {
        var order = new Order();
        order.Username = User.Identity.Name;
        MembershipUser currentUser = Membership.GetUser(User.Identity.Name, true /* userIsOnline */);
        storeDB.SaveChanges();
        var cart = ShoppingCart.GetCart(this.HttpContext);
        // Set up the ViewModel
        var viewModel = new CheckoutViewModel
        {
            CartItems = cart.GetCartItems(),
            CartTotal = cart.GetTotal(),
            Order = order
        };
        // Return the view
        return View(viewModel);
    }

控制器开机自检:操作结果

    [HttpPost]
    public ActionResult AddressAndPayment(CheckoutViewModel checkoutViewModel)
    {
        TryValidateModel(checkoutViewModel);
        try
        {
            checkoutViewModel.Order.Username = User.Identity.Name;
            checkoutViewModel.Order.OrderDate = DateTime.Now;
            storeDB.Orders.Add(checkoutViewModel.Order);
            // Error occurs on the following line if a custom data annotation validation attribute fails
            storeDB.SaveChanges();
            var cart = ShoppingCart.GetCart(this.HttpContext);
            cart.CreateOrder(checkoutViewModel.Order);
            storeDB.SaveChanges();
            return RedirectToAction("Complete", new { id = checkoutViewModel.Order.OrderID });
        }
        catch
        {
            //Invalid - redisplay with errors
            return View(checkoutViewModel);
        }
    }

编辑 #1
@SoWeLie和我讨论了它,我已经解决了这个问题。
为了在将购物车项传递回视图之前重新冻结视图,我将以下代码添加到我的 HttpPost ActionResult 的捕获块中。 此代码基本上调用命中数据库以获取购物车所需信息的方法:

catch
        {
            //Invalid - redisplay with errors
            //The model is not valid, we need to redisplay the same view so that the user can fix the errors => fetch the cartItems
             var cart = ShoppingCart.GetCart(this.HttpContext);
             checkoutViewModel.CartItems = cart.GetCartItems();
             checkoutViewModel.CartTotal = cart.GetTotal();
            return View(checkoutViewModel);
        }

您不希望在隐藏字段中保留整个对象,例如 CartItem 列表。 尝试将所有数据作为帖子值携带是低效且难以维护的。 您的 cart 对象需要持久化在应用程序(如数据库)中的某个位置。 您还可以将购物车存储在用户的会话中。

实现

此目的的最佳方法是将用户的购物车保存在数据库中。 您可以通过多种方式执行此操作。 在不知道您的数据库模型的情况下,最简单的解决方案是将购物车及其项目添加到数据库中,然后获取购物车 ID 并将其存储在会话或 cookie 中(取决于您希望购物车保留多长时间)。

最新更新