如何解决一个模型可以通过多个编辑器模板传递吗?
| 我正在尝试使用编辑器模板显示视图模型,该模型在应用基础对象编辑器模板之前将模型包装在字段集中。 我的看法:@model Mvc3VanillaApplication.Models.ContactModel
@using (Html.BeginForm())
{
@Html.EditorForModel(\"Fieldset\")
}
使用字段集模板(Views / Shared / EditorTemplates / Fieldset.cshtml):
<fieldset>
<legend>@ViewData.ModelMetadata.DisplayName</legend>
@Html.EditorForModel()
</fieldset>
依次使用所有对象的基本模板(Views / Shared / EditorTemplates / Object.cshtml):
@foreach (var prop in ViewData.ModelMetadata.Properties.Where(x =>
x.ShowForEdit && !x.IsComplexType && !ViewData.TemplateInfo.Visited(x)))
{
@Html.Label(prop.PropertyName,prop.DisplayName)
@Html.Editor(prop.PropertyName)
}
无论如何,这就是我的意图。问题在于,尽管页面使用字段集和图例进行渲染,但没有应用对象模板,因此不会显示任何输入控件。
如果我更改视图以不指定\“ Fieldset \”模板,则使用Object模板渲染我模型的属性,因此不是找不到我的Object模板。
是否可以通过多个模板传递同一模型?
对于它的价值,视图模型如下所示:
namespace Mvc3VanillaApplication.Models
{
[System.ComponentModel.DisplayName(\"Contact Info\")]
public class ContactModel
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
}
解决方法
我实现了您所拥有的,并能够复制它。我在ѭ4中设置了一个断点,以便可以对其进行检查,我措手不及,意识到在使用fieldset模板时,它甚至没有击中对象模板。然后,我逐步浏览了fieldset模板,发现它正在很好地调用该模板,因此代码中一定发生了某些事情,从而阻止了它显示对象模板。
我打开了MVC3源代码,搜索
EditorForModel
,发现正确的功能。
public static MvcHtmlString EditorForModel(this HtmlHelper html) {
return MvcHtmlString.Create(TemplateHelpers.TemplateHelper(html,html.ViewData.ModelMetadata,String.Empty,null /* templateName */,DataBoundControlMode.Edit,null /* additionalViewData */));
}
显然不是,所以我在TemplateHelpers.TemplateHelper
上按了F12
,然后再次在单行呼叫中按了F12
,这使您领会了函数的实质。在这里,我发现这段简短的代码从TemplateHelpers.cs
的第214行开始:
// Normally this shouldn\'t happen,unless someone writes their own custom Object templates which
// don\'t check to make sure that the object hasn\'t already been displayed
object visitedObjectsKey = metadata.Model ?? metadata.RealModelType;
if (html.ViewDataContainer.ViewData.TemplateInfo.VisitedObjects.Contains(visitedObjectsKey)) { // DDB #224750
return String.Empty;
}
这些注释实际上在代码中,在这里,我们可以回答您的问题:一个模型可以通过多个编辑器模板传递吗?答案为否*。
话虽如此,对于这样的功能来说,这似乎是一个非常合理的用例,因此寻找替代品可能值得付出努力。我怀疑模板化的剃刀委托可以解决这种包装功能,因此我尝试了一下。
@{
Func<dynamic,object> fieldset = @<fieldset><legend>@ViewData.ModelMetadata.DisplayName</legend>@Html.EditorForModel()</fieldset>;
}
@using (Html.BeginForm())
{
//@Html.EditorForModel(\"Fieldset\")
//@Html.EditorForModel()
@fieldset(Model)
}
和中提琴!有效!我将由您自己决定将其实现为扩展方法(以及更多可重用的方法)。这是有关模板化剃刀代表的简短博客文章。
*从技术上讲,您可以重写此函数并编译自己的MVC3版本,但这可能比它的价值还要麻烦。当您发现定义了几百条路线时,tried13ѭ函数的运行速度很慢,因此我们尝试在事业项目中执行此操作。其余的库存在一个签名问题,我们认为现在不值得花时间进行维护并维护将来的MVC版本。
, 在第一个cshtml模板中,我们可以重新创建ViewData.TemplateInfo(并清除VisitedObjects列表)
var templateInfo = ViewData.TemplateInfo;
ViewData.TemplateInfo = new TemplateInfo
{
HtmlFieldPrefix = templateInfo.HtmlFieldPrefix,FormattedModelValue = templateInfo.FormattedModelValue
};
现在我们可以调用另一个具有相同模型的模板
@Html.DisplayForModel(\"SecondTemplate\")
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。