一个模型可以通过多个编辑器模板传递吗?

问题描述

| 我正在尝试使用编辑器模板显示视图模型,该模型在应用基础对象编辑器模板之前将模型包装在字段集中。 我的看法:
@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\")
    

相关问答

错误1:Request method ‘DELETE‘ not supported 错误还原:...
错误1:启动docker镜像时报错:Error response from daemon:...
错误1:private field ‘xxx‘ is never assigned 按Alt...
报错如下,通过源不能下载,最后警告pip需升级版本 Requirem...