Answers:
使用ViewContext
和查看RouteData
集合,以提取控制器元素和动作元素。但是我认为设置一些指示应用程序上下文的数据变量(例如“ editmode”或“ error”),而不是控制器/操作会减少视图和控制器之间的耦合。
在RC中,您还可以像这样提取动作方法名称等路线数据
ViewContext.Controller.ValueProvider["action"].RawValue
ViewContext.Controller.ValueProvider["controller"].RawValue
ViewContext.Controller.ValueProvider["id"].RawValue
ViewContext.Controller.ValueProvider.GetValue("action").RawValue
ViewContext.Controller.ValueProvider.GetValue("controller").RawValue
ViewContext.Controller.ValueProvider.GetValue("id").RawValue
ViewContext.Controller.RouteData.Values["action"]
ViewContext.Controller.RouteData.Values["controller"]
ViewContext.Controller.RouteData.Values["id"]
ViewContext.RouteData.Values["action"]
ViewContext.RouteData.Values["controller"]
ViewContext.RouteData.Values["id"]
ViewContext.Controller.ValueProvider.GetValue("action").RawValue
变体了
要在视图上获取当前ID:
ViewContext.RouteData.Values["id"].ToString()
获取当前控制器:
ViewContext.RouteData.Values["controller"].ToString()
ViewContext.RouteData.Values.ContainsKey(<key>)
。
我知道这是一个较旧的问题,但是我看到了,我认为您可能对替代版本感兴趣,而不是让视图处理检索完成其工作所需的数据。
我认为,更简单的方法是重写OnActionExecuting方法。传递给您的ActionExecutingContext包含ActionDescriptor成员,您可以使用该成员获取您要查找的信息,即ActionName,还可以访问ControllerDescriptor,其中包含ControllerName。
protected override void OnActionExecuting(ActionExecutingContext filterContext)
{
ActionDescriptor actionDescriptor = filterContext.ActionDescriptor;
string actionName = actionDescriptor.ActionName;
string controllerName = actionDescriptor.ControllerDescriptor.ControllerName;
// Now that you have the values, set them somewhere and pass them down with your ViewModel
// This will keep your view cleaner and the controller will take care of everything that the view needs to do it's job.
}
希望这可以帮助。如果有的话,至少它会为您提出的其他问题提供替代方案。
我看到了不同的答案,并想出了一个班级助手:
using System;
using System.Web.Mvc;
namespace MyMvcApp.Helpers {
public class LocationHelper {
public static bool IsCurrentControllerAndAction(string controllerName, string actionName, ViewContext viewContext) {
bool result = false;
string normalizedControllerName = controllerName.EndsWith("Controller") ? controllerName : String.Format("{0}Controller", controllerName);
if(viewContext == null) return false;
if(String.IsNullOrEmpty(actionName)) return false;
if (viewContext.Controller.GetType().Name.Equals(normalizedControllerName, StringComparison.InvariantCultureIgnoreCase) &&
viewContext.Controller.ValueProvider.GetValue("action").AttemptedValue.Equals(actionName, StringComparison.InvariantCultureIgnoreCase)) {
result = true;
}
return result;
}
}
}
因此,在View(或master / layout)中,您可以像这样使用它(Razor语法):
<div id="menucontainer">
<ul id="menu">
<li @if(MyMvcApp.Helpers.LocationHelper.IsCurrentControllerAndAction("home", "index", ViewContext)) {
@:class="selected"
}>@Html.ActionLink("Home", "Index", "Home")</li>
<li @if(MyMvcApp.Helpers.LocationHelper.IsCurrentControllerAndAction("account","logon", ViewContext)) {
@:class="selected"
}>@Html.ActionLink("Logon", "Logon", "Account")</li>
<li @if(MyMvcApp.Helpers.LocationHelper.IsCurrentControllerAndAction("home","about", ViewContext)) {
@:class="selected"
}>@Html.ActionLink("About", "About", "Home")</li>
</ul>
</div>
希望能帮助到你。
您可以从ViewContext的RouteData获取这些数据
ViewContext.RouteData.Values["controller"]
ViewContext.RouteData.Values["action"]
在MVC中,您应该向View提供所有数据,而不是让View收集自己的数据,因此您可以在控制器操作中设置CSS类。
ViewData["CssClass"] = "bold";
并从View的ViewData中选择此值
我对此表示投票2:
string currentActionName = ViewContext.RouteData.GetRequiredString("action");
和
string currentViewName = ((WebFormView)ViewContext.View).ViewPath;
您可以同时检索当前视图的物理名称和触发它的操作。在部分* .acmx页面中确定主机容器可能很有用。
扩展Dale Ragan的答案(他的重用示例),创建一个从Controller派生的ApplicationController类,然后使所有其他控制器从该ApplicationController类而不是Controller派生。
例:
public class MyCustomApplicationController : Controller {}
public class HomeController : MyCustomApplicationController {}
在新的ApplicationController上,使用以下签名创建一个名为ExecutingAction的属性:
protected ActionDescriptor ExecutingAction { get; set; }
然后在OnActionExecuting方法(根据Dale Ragan的回答)中,只需将ActionDescriptor分配给该属性,您就可以在任何控制器中随时使用它。
string currentActionName = this.ExecutingAction.ActionName;
在控制器中覆盖此功能
protected override void HandleUnknownAction(string actionName)
{ TempData["actionName"] = actionName;
View("urViewName").ExecuteResult(this.ControllerContext);
}