asp.net-mvc – 如何使用我的Error.cshtml视图中的过滤器放置到ViewBag中的数据?

前端之家收集整理的这篇文章主要介绍了asp.net-mvc – 如何使用我的Error.cshtml视图中的过滤器放置到ViewBag中的数据?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我有一个动作过滤器,负责将一些常见的信息放在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(“在空引用上无法执行运行时绑定”)。

看来即使我的动作过滤器在抛出原始异常之前成功设置了ViewBag中的值,当渲染我的Error.cshtml视图时,会使用一个带有空的ViewBag的新上下文。

有什么办法可以将动作过滤器创建的数据用于自定义错误视图吗?

解决方法

我通过添加另一个过滤器来提出自己的解决方案。
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);
        }
    }
}

这个过滤器需要通过调用PreserveViewDataOnExceptionFilter.Register()在Global.asax.cs Application_Start()方法中全局连接。

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

猜你在找的asp.Net相关文章