Answers:
从MVC2开始,您可以使用 ViewContext.RouteData.DataTokens["area"]
ASP.NET Core 1.0
而不是MVC6
.. :-)
在ASP.NET Core 1.0中,该值位于
ViewContext.RouteData.Values [“ area”];
pageModel.RouteData.Values.TryGetValue("area", out object area)
应该使用)
我只是为此写了一个日志条目,您可以访问它以获取更多详细信息,但是我的答案是创建一个扩展方法,如下所示。
关键因素是您从.DataTokens中拉出MVC区域,并从RouteData的.Values中拉出控制器/动作。
public static MvcHtmlString TopMenuLink(this HtmlHelper htmlHelper, string linkText, string controller, string action, string area, string anchorTitle)
{
var urlHelper = new UrlHelper(htmlHelper.ViewContext.RequestContext);
var url = urlHelper.Action(action, controller, new { @area = area });
var anchor = new TagBuilder("a");
anchor.InnerHtml = HttpUtility.HtmlEncode(linkText);
anchor.MergeAttribute("href", url);
anchor.Attributes.Add("title", anchorTitle);
var listItem = new TagBuilder("li");
listItem.InnerHtml = anchor.ToString(TagRenderMode.Normal);
if (CheckForActiveItem(htmlHelper, controller, action, area))
listItem.GenerateId("menu_active");
return MvcHtmlString.Create(listItem.ToString(TagRenderMode.Normal));
}
private static bool CheckForActiveItem(HtmlHelper htmlHelper, string controller, string action, string area)
{
if (!CheckIfTokenMatches(htmlHelper, area, "area"))
return false;
if (!CheckIfValueMatches(htmlHelper, controller, "controller"))
return false;
return CheckIfValueMatches(htmlHelper, action, "action");
}
private static bool CheckIfValueMatches(HtmlHelper htmlHelper, string item, string dataToken)
{
var routeData = (string)htmlHelper.ViewContext.RouteData.Values[dataToken];
if (routeData == null) return string.IsNullOrEmpty(item);
return routeData == item;
}
private static bool CheckIfTokenMatches(HtmlHelper htmlHelper, string item, string dataToken)
{
var routeData = (string)htmlHelper.ViewContext.RouteData.DataTokens[dataToken];
if (dataToken == "action" && item == "Index" && string.IsNullOrEmpty(routeData))
return true;
if (dataToken == "controller" && item == "Home" && string.IsNullOrEmpty(routeData))
return true;
if (routeData == null) return string.IsNullOrEmpty(item);
return routeData == item;
}
然后,您可以如下实现它:
<ul id="menu">
@Html.TopMenuLink("Dashboard", "Home", "Index", "", "Click here for the dashboard.")
@Html.TopMenuLink("Courses", "Home", "Index", "Courses", "List of our Courses.")
</ul>
area
令牌,Values
但需要查找DataTokens
...。不知道为什么。
我创建了一个扩展方法,用于RouteData
返回当前区域名称。
public static string GetAreaName(this RouteData routeData)
{
object area;
if (routeData.DataTokens.TryGetValue("area", out area))
{
return area as string;
}
return null;
}
因为RouteData
两者都可用,ControllerContext
并且ViewContext
可以在控制器和视图中访问它。
测试也很容易:
[TestFixture]
public class RouteDataExtensionsTests
{
[Test]
public void GetAreaName_should_return_area_name()
{
var routeData = new RouteData();
routeData.DataTokens.Add("area", "Admin");
routeData.GetAreaName().ShouldEqual("Admin");
}
[Test]
public void GetAreaName_should_return_null_when_not_set()
{
var routeData = new RouteData();
routeData.GetAreaName().ShouldBeNull();
}
}
RouteData.DataTokens
由于始终在内部初始化,因此无需检查是否为null。
我知道这很旧,但是在使用ActionFilter之类的过滤器时,上下文也不容易为您提供区域信息。
可以在以下代码中找到它:
var routeData = filterContext.RequestContext.RouteData;
if (routeData.DataTokens["area"] != null)
area = routeData.DataTokens["area"].ToString();
因此,将在覆盖上传递filterContext,并在RequestContext下找到正确的RouteData。在基础级别上有一个RoutData,但是DataTokens的字典中没有该区域。
我知道这是一个非常老的帖子,但是我们可以使用与DataTokens完全相同的方式使用Values属性
Url.RequestContext.RouteData.Values [“ action”]为我工作。
我不知道为什么,但是接受的答案不起作用。它用例如返回null(也许关于mvc,我使用.net core)
我总是调试变量并从中获取数据。
试试这个。这个对我有用
var area = ViewContext.RouteData.Values["area"]
详细的逻辑示例
Layout = ViewContext.RouteData.Values["area"] == null ? "_LayoutUser" : "_LayoutAdmin";