我如何在Error.cshtml视图中使用由过滤器放置到ViewBag中的数据?

18

我有一个Action Filter,它负责将一些常用信息放入ViewBag中,以供在共享的_Layout.cshtml文件中的所有视图中使用。

public class ProductInfoFilterAttribute : ActionFilterAttribute
{
    public override void
    OnActionExecuting(ActionExecutingContext filterContext)
    {
        //  build product info
        //  ... (code omitted)

        dynamic viewBag = filterContext.Controller.ViewBag;
        viewBag.ProductInfo = info;
    }
}
在共享的_Layout.cshtml文件中,我使用已经存储在ViewBag中的信息。
...
@ViewBag.ProductInfo.Name
...
如果在处理控制器操作时出现异常,标准的HandleErrorAttribute应该显示我的共享Error.cshtml视图,在我引入上面的操作筛选器并开始在_Layout.cshtml中使用ViewBag的新值之前,这是有效的。现在我得到的是标准的ASP.Net运行时错误页面,而不是自定义的Error.cshtml视图。
我发现问题在于在渲染错误视图时,在_Layout.cshtml中使用ViewBag.ProductInfo.Name会抛出一个RuntimeBinderException("无法对null引用执行运行时绑定")。
看起来即使我的操作筛选器在原始异常抛出之前成功地将值设置在ViewBag中,但在呈现我的Error.cshtml视图时仍然会使用一个空的ViewBag新上下文。
有没有办法让操作筛选器创建的数据可用于自定义错误视图?
1个回答

12

我通过添加另一个过滤器,想出了自己的解决方案。

public class PreserveViewDataOnExceptionFilter : IExceptionFilter
{
    public void
    OnException(ExceptionContext filterContext)
    {
        //  copy view data contents from controller to result view
        ViewResult viewResult = filterContext.Result as ViewResult;
        if ( viewResult != null )
        {
            foreach ( var value in filterContext.Controller.ViewData )
            {
                if ( ! viewResult.ViewData.ContainsKey(value.Key) )
                {
                    viewResult.ViewData[value.Key] = value.Value;
                }
            }
        }
    }

    public static void
    Register()
    {
        FilterProviders.Providers.Add(new FilterProvider());
    }

    private class FilterProvider : IFilterProvider
    {
        public IEnumerable<Filter>
        GetFilters(ControllerContext controllerContext, ActionDescriptor actionDescriptor)
        {
            //  attach filter as "first" for all controllers / actions; note: exception filters run in reverse order
            //  so this really causes the filter to be the last filter to execute
            yield return new Filter(new PreserveViewDataOnExceptionFilter(), FilterScope.First, null);
        }
    }
}

需要在Global.asax.cs的Application_Start()方法中全局挂钩此过滤器,通过调用PreserveViewDataOnExceptionFilter.Register()来实现。

我所做的是设置一个新的异常过滤器,它在HandleErrorAttribute过滤器运行后最后运行,并将由控制器抛出异常时可用的ViewData集合的内容复制到HandleErrorAttribute过滤器创建的结果中。


1
这对于ViewData非常有效,我喜欢这段代码,但是你有没有相应的ViewBag代码? - Nick Coad
filterContext.Controller.ViewData 包含了 Model 和 ViewBag,据我所见,你可以将它们全部传递,或者只传递需要的部分。 - Peter Kerr

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接