我的应用程序中有一对视图,它们都为我的一个模型项显示相同的编辑器模板;在两个视图(“添加”和“编辑”)中,“编辑”工作正常,但当我的控制器操作处理帖子时,“添加”为模型返回null。
我发现如果我给“添加”视图一个自定义ViewModel并调用Html.EditorFor(p => p.PageContent)
而不是简单地调用整个Model对象上的EditorFor() - Html.EditorFor(p => p)
,那么表单将返回正确的,非null模型,但会产生与我的客户端脚本和控件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) {}
在你哭“重复”之前
这个问题确实与this one有关,但这个问题的目的是针对我遇到的具体问题,而不是那里提出的更普遍的问题。
答案 0 :(得分:14)
我追踪了这个问题,这是一个相当有趣的问题。
当DefaultModelBinder尝试解析模型项时,它所做的第一件事就是检查数据中是否有任何带前缀的字段;它通过检查以模型对象的名称开头的任何表单项来实现这一点(如果你问我,这似乎非常随意)。如果找到任何“前缀”字段,则会导致调用不同的绑定逻辑。
ASP.NET MVC 2预览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);
}
我定义的用于处理Add操作的控制器操作定义了一个名为“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模型项。我不确定为什么编辑视图也没有显示这种行为,但无论哪种方式我都追查了问题的根源。
在我看来,这个问题非常接近“bug”状态。我的视图最初与ViewModel一起工作是有道理的,因为“内容”的前缀是“PageContent_”,但是像这样的核心框架功能/ bug应该不会得到解决恕我直言。