【问题标题】:Model binding with nested child models and PartialViews in ASP.NET MVCASP.NET MVC 中与嵌套子模型和 PartialViews 的模型绑定
【发布时间】:2011-01-28 13:59:48
【问题描述】:

我有以下类型和类:

namespace MVC.Models

public class Page 
{
   public EditableContent Content {get; set; }
}

public class EditableContent
{
    public TemplateSection SidebarLeft {get; set; }
    public TemplateSection SidebarRight {get; set; }
}

我想在我的Edit.aspx 视图中编辑Page 实例。因为EditableContent 也附加到其他模型,所以我有一个名为ContentEditor.ascxPartialView,它是强类型的,并采用EditableContent 的实例并呈现它。

渲染部分一切正常,但是当我发布时 - 我的 ContentEditor 中的所有内容都未绑定 - 这意味着 Page.Contentnull

在 PartialView 上,我使用强类型的 Html Helpers 来执行此操作:

<%= Html.HiddenFor(m => m.TemplateId) %>

但是由于ContentEditor.ascx 呈现的表单上的输入元素没有将Content 前缀添加到其id 属性 - 值未绑定到Page

我尝试使用松散类型的助手来克服这个问题:

<%= Html.Hidden("Content.TemplateId", Model.TemplateId) %>

当我处理一个List&lt;T&gt; 的属性时,它会变得非常难看。然后我必须手动呈现集合索引。

我应该将 Page 和 EditableContent 作为参数放入控制器操作吗?:

public ActionResult Edit(Page page, EditableContent content) { ... }

我错过了什么?

【问题讨论】:

    标签: c# .net asp.net-mvc model model-binding


    【解决方案1】:

    我建议你使用EditorFor 助手

    型号:

    public class EditableContent
    {
        public string SidebarLeft { get; set; }
        public string SidebarRight { get; set; }
    }
    
    public class Page
    {
        public EditableContent Content { get; set; }
    }
    

    Views/Home/Index.aspx:

    <%@ Page Language="C#" MasterPageFile="~/Views/Shared/Site.Master" Inherits="System.Web.Mvc.ViewPage<ToDD.Models.Page>" %>
    
    <asp:Content ID="Content1" ContentPlaceHolderID="TitleContent" runat="server">
        Home Page
    </asp:Content>
    
    <asp:Content ID="Content2" ContentPlaceHolderID="MainContent" runat="server">
    <% using (Html.BeginForm()) { %>
        <%-- 
        This is the important part: It will look for 
        Views/Shared/EditorTemplates/EditableContent.ascx
        and render it. You could also specify a prefix
        --%>
        <%= Html.EditorFor(page => page.Content, "Content") %>
        <input type="submit" value="create" />
    <% } %>
    </asp:Content>
    

    Views/Shared/EditorTemplates/EditableContent.ascx:

    <%@ Control Language="C#" Inherits="System.Web.Mvc.ViewUserControl<ToDD.Models.EditableContent>" %>
    
    <%= Html.TextBoxFor(m => m.SidebarLeft) %>
    <br/>
    <%= Html.TextBoxFor(m => m.SidebarRight) %>
    

    最后是 Controller/HomeController:

    public class HomeController : Controller
    {
        public ActionResult Edit()
        {
            var page = new Page
            {
                Content = new EditableContent
                {
                    SidebarLeft = "left",
                    SidebarRight = "right"
                }
            };
            return View(page);
        }
    
        [HttpPost]
        public ActionResult Edit(Page page)
        {
            return View(page);
        }
    }
    

    【讨论】:

    • 它有效,伙计!谢谢。我假设 EditorFor 负责将所有的脚手架连接在一起。
    • 我在我的一个项目上做过这个,但你能解释一下为什么 Html.RenderPartial("_blah" , Model.Blah) 不工作达林?因为这就是我们以前的情况。
    • @GONeale,它不起作用,因为你传递了 Model.Blah,所以在部分内部你是在 Blah 的上下文中,而不是在 Model => 你在这个使用的所有助手的上下文中部分生成错误的输入名称。编辑器模板不同 => 它们考虑到父上下文并生成在它们内部使用的帮助器将生成正确的输入名称。只需查看您的标记,您就会立即看到不同之处。要使默认模型绑定器正常工作,您必须遵守约定。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-09-27
    • 1970-01-01
    • 2014-07-10
    • 2014-01-19
    相关资源
    最近更新 更多