Comment obtenez-vous le nom de la zone actuelle dans la vue ou le contrôleur?
Y a-t-il quelque chose comme ViewContext.RouteData.Values["controller"]
pour les zones?
Comment obtenez-vous le nom de la zone actuelle dans la vue ou le contrôleur?
Y a-t-il quelque chose comme ViewContext.RouteData.Values["controller"]
pour les zones?
Réponses:
À partir de MVC2, vous pouvez utiliser ViewContext.RouteData.DataTokens["area"]
ASP.NET Core 1.0
place de MVC6
.. :-)
Vous pouvez l'obtenir à partir du contrôleur en utilisant:
ControllerContext.RouteData.DataTokens["area"]
Dans ASP.NET Core 1.0, la valeur se trouve dans
ViewContext.RouteData.Values ["zone"];
pageModel.RouteData.Values.TryGetValue("area", out object area)
devrait être utilisé à la place)
Je viens d'écrire une entrée de journal ab à ce sujet , vous pouvez le visiter pour plus de détails, mais ma réponse a été de créer une méthode d'extension, illustrée ci-dessous.
Le déclencheur clé était que vous extrayiez la zone MVC des .DataTokens et le contrôleur / l'action des .Values de RouteData.
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;
}
Ensuite, vous pouvez l'implémenter comme ci-dessous:
<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
jeton dans le Values
mais je devais regarder dans le DataTokens
... aucune idée pourquoi.
J'ai créé une méthode d'extension pour RouteData
qui renvoie le nom de la zone actuelle.
public static string GetAreaName(this RouteData routeData)
{
object area;
if (routeData.DataTokens.TryGetValue("area", out area))
{
return area as string;
}
return null;
}
Depuis RouteData
est disponible sur les deux ControllerContext
et ViewContext
il est accessible dans votre contrôleur et vos vues.
Il est également très facile de tester:
[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();
}
}
Il n'est pas nécessaire de vérifier si RouteData.DataTokens
est nul car il est toujours initialisé en interne.
MVC Futures a une méthode AreaHelpers.GetAreaName (). Cependant, soyez prudent si vous utilisez cette méthode. L'utilisation de la zone actuelle pour prendre des décisions d'exécution concernant votre application peut entraîner un code difficile à déboguer ou non sécurisé.
this.GetName()
et pour l'utilisation actuelle de la méthodeMethodBase.GetCurrentMethod().Name
Je sais que c'est vieux, mais aussi, quand dans un filtre comme ActionFilter, le contexte ne vous fournit pas facilement les informations de zone.
Il peut être trouvé dans le code suivant:
var routeData = filterContext.RequestContext.RouteData;
if (routeData.DataTokens["area"] != null)
area = routeData.DataTokens["area"].ToString();
Ainsi, le filterContext est transmis lors du remplacement et le RouteData correct se trouve sous RequestContext. Il y a un RoutData au niveau de base, mais les DataTokens n'ont PAS la zone dans leur dictionnaire.
Je sais que c'est un article très très ancien, mais nous pouvons utiliser la propriété Values exactement de la même manière que les DataTokens
Url.RequestContext.RouteData.Values ["action"] a fonctionné pour moi.
Je ne sais pas pourquoi mais la réponse acceptée ne fonctionne pas. Il renvoie null avec par exemple (peut-être à propos de mvc, j'utilise .net core)
Je débogue toujours la variable et j'y extrait des données.
Essaye ça. Ça marche pour moi
var area = ViewContext.RouteData.Values["area"]
Exemple logique détaillé
Layout = ViewContext.RouteData.Values["area"] == null ? "_LayoutUser" : "_LayoutAdmin";