4

我的应用程序中有一对视图,它们都为我的一个模型项显示相同的编辑器模板;在两个视图(“添加”和“编辑”)中,“编辑”工作正常,但是当我的控制器操作处理帖子时,“添加”为模型返回 null。

我发现如果我给“添加”视图一个自定义 ViewModel 并调用Html.EditorFor(p => p.PageContent)而不是简单地调用整个 Model 对象上的 EditorFor() Html.EditorFor(p => p),那么表单会返回正确的非空模型,但这会产生与我的客户端脚本和控件 ID(因为现在所有字段都以“PageContent_”为前缀)。我在整个应用程序的几个不同地方使用了相同的编辑器模板技术,而其他人都没有表现出对 ViewModel 的这种奇怪的依赖。

有没有其他人遇到过类似的问题?

编辑视图

<%@ Page Title="" Language="C#" MasterPageFile="~/Areas/Admin/Views/Shared/Site.Master"
Inherits="System.Web.Mvc.ViewPage<PageContent>" %>

<% using (Html.BeginForm())
   { %>
<%=Html.Hidden("PageID", Model.Page.ID) %>
<%=Html.EditorFor(p => p)%>
<input type="submit" name="btnSave" value="Save" />
<input type="submit" name="btnCancel" value="Cancel" class="cancel" />
<% }

行动(工作)

[HttpPost, ValidateInput(false)]
public ActionResult EditContent(int id, FormCollection formCollection) {}

添加视图

<%@ Page Title="" Language="C#" MasterPageFile="~/Areas/Admin/Views/Shared/Site.Master"
Inherits="System.Web.Mvc.ViewPage<PageContent>" %>

<% using (Html.BeginForm())
   { %>
<%=Html.Hidden("PageID", ViewData["PageID"]) %>
<%=Html.EditorFor(p => p)%>
<input type="submit" name="btnSave" value="Save" />
<input type="submit" name="btnCancel" value="Cancel" class="cancel" />
<% } %>

行动(失败)

// content is ALWAYS null
[HttpPost, ValidateInput(false)]
public ActionResult AddContent(PageContent content, FormCollection formCollection) {}

在你哭“重复”之前

这个问题确实与这个有关,但这个问题旨在针对我遇到的具体问题,而不是那里提出的更一般的问题。

4

1 回答 1

14

我找到了这个问题,这是一个相当有趣的问题。

当 DefaultModelBinder 尝试解析模型项时,它首先要做的事情之一是检查绑定的数据中是否有任何前缀字段;它通过检查以模型对象名称开头的任何表单项来做到这一点(如果你问我,这似乎非常随意)。如果找到任何“前缀”字段,则会导致调用不同的绑定逻辑。

ASP.NET MVC 2 Preview 2 BindModel() 源码

public virtual object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext) {
    if (bindingContext == null) {
        throw new ArgumentNullException("bindingContext");
    }
    bool performedFallback = false;
    if (!String.IsNullOrEmpty(bindingContext.ModelName) && !DictionaryHelpers.DoesAnyKeyHavePrefix(bindingContext.ValueProvider, bindingContext.ModelName)) {
        // We couldn't find any entry that began with the prefix. If this is the top-level element, fall back
        // to the empty prefix.
        if (bindingContext.FallbackToEmptyPrefix) {
             /* omitted for brevity */
            };
            performedFallback = true;
        }
        else {
            return null;
        }
    }

    // Simple model = int, string, etc.; determined by calling TypeConverter.CanConvertFrom(typeof(string))
    // or by seeing if a value in the request exactly matches the name of the model we're binding.
    // Complex type = everything else.
    if (!performedFallback) {
       /* omitted for brevity */
    }
    if (!bindingContext.ModelMetadata.IsComplexType) {
        return null;
    }
    return BindComplexModel(controllerContext, bindingContext);
}

我为处理添加操作而定义的控制器操作定义了一个名为“content”的 PageContent 项,并且在我的域 PageContent 中有一个名为“Content”的属性,它与“content”的模型名称“匹配”,因此导致 DefaultModelBinder 假设我有一个前缀值,而实际上它只是 PageContent 的一个成员。通过更改签名-

从:

[HttpPost, ValidateInput(false)]
public ActionResult AddContent(PageContent content, FormCollection formCollection) {}

到:

[HttpPost, ValidateInput(false)]
public ActionResult AddContent(PageContent pageContent, FormCollection formCollection) {}

DefaultModelBinder 再次能够正确绑定到我的 PageContent 模型项。我不确定为什么编辑视图也没有显示这种行为,但无论哪种方式我都已经找到了问题的根源。

在我看来,这个问题非常接近“错误”状态。我的视图最初与 ViewModel 一起工作是有道理的,因为“内容”的前缀是“PageContent_”,但是像这样的核心框架功能/错误不应该在恕我直言。

于 2009-11-30T20:16:27.103 回答