2009-10-14 110 views
5

我有一个控制器是从一个基础控制器继承,我想知道如何利用基础控制器的所有逻辑,但返回一个不同于基础控制器使用的视图。如何继承ASP.NET MVC控制器并只更改视图?

基本控制器填充一个模型对象并将该模型对象传递给它的视图,但我不知道如何在子控制器中访问该模型对象,以便我可以将它传递给子控制器的视图。从我的应用程序

+1

您是否尝试从基本控制器继承并准备视图? – LukLed 2009-10-14 15:31:08

回答

1

基于此线程给出的反馈,我实现了一个类似Antony Koch提出的解决方案。

取而代之的是使用抽象方法,我使用了一个具体的,虚拟的GetIndex方法,这样我就可以为它的基础控制器放置逻辑。

public class SalesController : Controller 
{ 
    // Index view method and model 
    public virtual ActionResult GetIndex() 
    { 
     return View("Index", IndexModel); 
    } 
    protected TestModel IndexModel { get; set; } 

    public virtual ActionResult Index() 
    { 
     ViewData["test"] = "Set in base."; 

     IndexModel = new TestModel(); 
     IndexModel.Text = "123"; 

     return GetIndex(); 
    } 

    [AcceptVerbs(HttpVerbs.Post)] 
    public virtual ActionResult Index(TestModel data, FormCollection form) 
    { 
     TryUpdateModel(data, form.ToValueProvider()); 
     IndexModel = data; 

     return GetIndex(); 
    } 
} 

// This class will need to be in a different namespace or named differently than the 
// parent controller 
public class SalesController : MyApp.Controllers.BaseControllers.SalesController 
{ 
    // Index view method and model 
    public override ActionResult GetIndex() 
    { 
     return View("ClientIndex", IndexModel); 
    } 

    public override ActionResult Index() 
    { 
     return base.Index(); 
    } 

    [AcceptVerbs(HttpVerbs.Post)] 
    public override ActionResult Index(TestModel data, FormCollection form) 
    { 
     return base.Index(data, form); 
    } 
} 
+0

我面临着同样的问题,任何人都可以帮助我吗? [我的问题](http://stackoverflow.com/questions/12720611/cant-add-controller-which-uses-class-which-is-inherited-from-other-class) – user1668543 2012-10-05 06:25:04

3

样品:

基类:

public abstract class BaseTableController<T,TU> : BaseController where TU : IGenericService<T>,IModelWrapperService 
{ 
    protected readonly TU _service; 

    public BaseTableController(TU service) 
    { 
     _service = service; 
     _service.ModelWrapper = new ControllerModelStateWrapper(ModelState); 
    } 


    public ActionResult Index() 
    { 
     return View(_service.List()); 
    } 

继承:

public class SeverityController : BaseTableController<Severity, ISeverityService> 
{ 
    public SeverityController(ISeverityService service) 
     : base(service) 
    { 
    } 

    //NO CODE INSIDE 
} 

SeverityController.Index()导致查看/严重性/的Index.aspx 。只需准备视图。严重性是我的错误跟踪应用程序中的一个词典。每个字典都有类似的逻辑,所以我可以分享一些代码。

0

我最终只是在基础控制器 - viewName上添加了一个额外的参数。

似乎工作得很好。

我错过了任何主要的缺点?

public class SalesController : Controller 
{ 
    public virtual ActionResult Index(string viewName) 
    { 
     ViewData["test"] = "Set in base."; 

     TestModel model = new TestModel(); 
     model.Text = "123"; 

     return String.IsNullOrEmpty(viewName) ? View(model) : View(viewName, model); 
    } 

    [AcceptVerbs(HttpVerbs.Post)] 
    public virtual ActionResult Index(TestModel data, FormCollection form, string viewName) 
    { 
     TryUpdateModel(data, form.ToValueProvider()); 
     return String.IsNullOrEmpty(viewName) ? View(data) : View(viewName, data); 
    } 
} 

public class SalesController : MyApp.Controllers.BaseControllers.SalesController 
{ 
    public override ActionResult Index(string viewName) 
    { 
     return base.Index("ClientIndex"); 
    } 

    [AcceptVerbs(HttpVerbs.Post)] 
    public override ActionResult Index(TestModel data, FormCollection form, string viewName) 
    { 
     return base.Index(data, form, "ClientIndex"); 
    } 
} 
+1

我担心的是用户可以通过http将参数发送到控制器中的ActionResult。如果用户能够在不覆盖“索引”的控制器上调用索引,从而能够传递具有意外结果的视图名称,就会冒风险。不太可能,但仍然有可能。 – 2009-10-14 16:00:35

+1

我不喜欢这个解决方案。您不必指定视图名称。想象一下你有一个BaseController和两个继承 - ChildOneController和ChildTwoController。如果您调用〜/ ChildOne/Index,它会从〜/ ChildOne/Index/Index.aspx中查看。如果您调用/ ChildTwo/Index,它会从〜/ ChildTwo/Index/Index.aspx中查看。视图名称相同,但是来自不同的地方。 – LukLed 2009-10-14 16:09:00

+0

我面临着同样的问题,任何人都可以帮助我吗? [我的问题](http://stackoverflow.com/questions/12720611/cant-add-controller-which-uses-class-which-is-inherited-from-other-class) – user1668543 2012-10-05 07:13:05

0
public class BaseController : Controller { 
protected BaseController() {} 

public ActionResult Index() 
{ 
    return GetIndex(); 
} 

public abstract ActionResult GetIndex(); } 

public class MyController : BaseController { 
public MyController() {} 

public override GetIndex() 
{ 
    return RedirectToAction("Cakes","Pies"); 
} 

}

只需使用抽象给你打电话从子类需要的位。

+0

我做了类似的事情。看到我的新答案。我希望避免为每个控制器操作创建辅助方法,但我想这是最好的方法。 – 2009-10-14 17:30:17

7

有几点。如果您知道这就是您要返回的所有内容,则可以将您的返回值输入为ViewResult。然后你可以从重写的实现中询问这个值。更重要的是,根据MVC v1源代码,调用View(object)只需在控制器上设置ViewData.Model,然后构造一个ViewResult。

Controller.cs:440

protected internal ViewResult View(object model) { 
    return View(null /* viewName */, null /* masterName */, model); 
} 

Controller.cs:456

protected internal virtual ViewResult View(string viewName, string masterName, object model) { 
    if (model != null) { 
     ViewData.Model = model; 
    } 

    return new ViewResult { 
     ViewName = viewName, 
     MasterName = masterName, 
     ViewData = ViewData, 
     TempData = TempData 
    }; 
} 

因此,所有你需要做的是调用基方法,并调用视图(串)。

namespace BaseControllers 
{ 
    public class CoolController 
    { 
     public virtual ViewResult Get() 
     { 
      var awesomeModel = new object(); 
      return View(awesomeModel); 
     } 
    } 
} 

public class CoolController : BaseControllers.CoolController 
{ 
    public override ViewResult Get() 
    { 
     var ignoredResult = base.Get(); 
     // ViewData.Model now refers to awesomeModel 
     return View("NotGet"); 
    } 
} 

当然,你浪费CPU周期来构造你忽略的ViewResult。因此,你可以这样做:

public class CoolController : BaseControllers.CoolController 
{ 
    public override ViewResult Get() 
    { 
     var baseResult = base.Get(); 
     baseResult.ViewName = "NotGet"; 
     return baseResult; 
    } 
} 

如果你的基本控制器返回的ActionResult,你就必须改变之前VIEWNAME把它转换为的ViewResult。