在ASP.NET MVC中创建部分视图的控制器

65

我如何为部分视图创建一个独立的控制器和模型? 我想能够在网站的任何位置放置这个部分视图,所以它需要自己的控制器。 我目前是这样呈现部分视图的

@Html.Partial("_Testimonials")

http://www.c-sharpcorner.com/article/html-action-and-html-renderaction-in-Asp-Net-mvc/ - Iman
6个回答

120

为什么不使用Html.RenderAction()呢?

这样,你可以在任何控制器中放置以下代码(甚至可以为此创建一个新的控制器):

[ChildActionOnly]
public ActionResult MyActionThatGeneratesAPartial(string parameter1)
{
    var model = repository.GetThingByParameter(parameter1);
    var partialViewModel = new PartialViewModel(model);
    return PartialView(partialViewModel); 
}

那么你可以创建一个新的部分视图,并让PartialViewModel成为其继承者。

对于 Razor,视图中的代码块应该像这样:

@{ Html.RenderAction("Index", "Home"); }

对于WebFormsViewEngine,它的代码应该是这样的:

<% Html.RenderAction("Index", "Home"); %>

4
使用RenderAction的唯一问题是它会发起第二个请求,而RenderPartial则是在同一个初始请求中完成。大多数情况下并不重要,但值得注意。请参见https://dev59.com/CXRB5IYBdhLWcg3wF0LH以了解一些背景知识。 - jim tollan
在vbhtml中,我会收到错误消息“表达式不产生值” ,该错误出现在 @Html.RenderAction("MyActionThatGeneratesAPartial", "Home") - KyleMit
@kylemit,你的代码是这样的吗?@{ Html.RenderAction("Index", "Home"); } - George Stocker
@GeorgeStocker,不对。诀窍是将其放入代码块中 - KyleMit
1
这对我来说是完美的解决方案。我正在使用return PartialView()来呈现一个包含站点标题下拉列表的Partial View,就像这样在_Layout View中:PartialView("_AccountsDDL", model); - landsteven

10

如果我是你,我会简单地创建一个带有单一操作的新控制器,然后使用 RenderAction 来替代 Partial:

// Assuming the controller is named NewController
@{Html.RenderAction("ActionName", 
                     "New", 
                      new { routeValueOne = "SomeValue" });
}

3
在您的代码片段中,使用您在答案中描述的RenderAction会更好吗? - Martin
不得不说,这是我迄今为止找到的唯一一个部分封装其自身功能的示例,因此我可以在任何地方放置一行代码,而无需修改其他视图/控制器等。 - djack109

10

它不需要自己的控制器。您可以使用

@Html.Partial("../ControllerName/_Testimonials.cshtml")

这允许你从任何页面呈现这个局部视图。只需确保相对路径是正确的。


4
为什么会被投票反对?假设需要一个独立的控制器才能在网站的任何页面上呈现部分内容,但这是错误的。您可以从任何控制器呈现部分视图。这是一个有效的答案。 - Chris Woolum
3
只是猜测,谁给你点了踩可能想要一个 ViewModel 与部分视图配合使用,并将其创建留给单独的专用控制器。您的示例在没有 ViewModel 的情况下运作良好,但强制当前(所有)控制器创建 / 填充 ViewModel 并将其传递给部分视图。 - Per Hornshøj-Schierbeck
不要忘记扩展名:Html.Partial("../ControllerName/_Testimonials.cshtml")。 - JoshYates1980

4
最重要的是,创建的动作必须返回部分视图,请参见以下内容。
public ActionResult _YourPartialViewSection()
{
    return PartialView();
}

3

您不需要控制器,而在使用 .Net 5 (MVC 6) 时,您可以异步渲染局部视图。

@await Html.PartialAsync("_LoginPartial")

或者

@{await Html.RenderPartialAsync("PartialName");}

如果您正在使用 .NET Core 2.1 或更高版本,您可以直接使用partial标签助手:

<partial name="Shared/_ProductPartial.cshtml"
         for="Product" />

0

Html.Action是一种设计不太好的技术。 因为在您的页面控制器中,您无法在部分控制器中接收计算结果。数据流只有页面控制器=>部分控制器。

为了更接近WebForm UserControl(*.ascx),您需要:

  1. 创建一个页面模型和一个部分模型

  2. 将您的部分模型作为属性放置在您的页面模型中

  3. 在页面的视图中使用Html.EditorFor(m => m.MyPartialModel)
  4. 创建一个适当的部分视图
  5. 创建一个非常类似于这里多次描述的Child Action Controller的类。但它只是一个类(继承自Object而不是Controller)。让我们把它命名为MyControllerPartial。MyControllerPartial只会知道部分模型。
  6. 在您的页面控制器中使用您的MyControllerPartial。将model.MyPartialModel传递给MyControllerPartial
  7. 注意MyControllerPartial中的正确前缀。例如:ModelState.AddError("MyPartialModel." + "SomeFieldName", "Error")
  8. 在MyControllerPartial中,您可以进行验证并实现与此部分模型相关的其他逻辑

在这种情况下,您可以像这样使用它:

public class MyController : Controller
{
    ....
    public MyController()
    {
    MyChildController = new MyControllerPartial(this.ViewData);
    }

    [HttpPost]
    public ActionResult Index(MyPageViewModel model)
    {
    ...
    int childResult = MyChildController.ProcessSomething(model.MyPartialModel);
    ...
    }
}

附言: 在第三步中,您可以使用Html.Partial("PartialViewName", Model.MyPartialModel, <clone_ViewData_with_prefix_MyPartialModel>)。更多细节请参见ASP.NET MVC部分视图:输入名称前缀


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接