gpt4 book ai didi

.net - 用于属性的 ASP.NET MVC 编辑器模板

转载 作者:行者123 更新时间:2023-12-04 01:00:50 26 4
gpt4 key购买 nike

通常我通过@Html.RenderModel 渲染我的表单,但这次我有一个复杂的渲染逻辑,我手动渲染它。我决定为一个属性创建一个编辑器模板。这是代码(从默认对象编辑器模板实现复制粘贴):

<%@ Control Language="C#" Inherits="System.Web.Mvc.ViewUserControl" %>
<% var modelMetadata = ViewData.ModelMetadata; %>
<% if (modelMetadata.HideSurroundingHtml) { %>
<%= Html.Editor(modelMetadata.PropertyName) %>
<% } else { %>
<% if (!String.IsNullOrEmpty(Html.Label(modelMetadata.PropertyName).ToHtmlString())) { %>
<div class="editor-label"><%= Html.Label(modelMetadata.PropertyName) %></div>
<% } %>
<div class="editor-field">
<%= Html.Editor(modelMetadata.PropertyName) %>
<%= Html.ValidationMessage(modelMetadata.PropertyName) %>
</div>
<% } %>

这是我如何使用它:
@Html.EditorFor(x => x.SomeProperty, "Property") //"Property" is template above

但它不起作用:无论 DisplayName 和编辑器都不会渲染标签(在 Watches Html.Editor(modelMetadata.PropertyName 显示空字符串)。我做错了什么?

最佳答案

您正在从您的编辑器中调用编辑器。正如@RPM1984 在 this 的评论中改写 @darin-dmitrov 一样答案:对于给定的类型,在给定的 View 特定上下文中,您只能在运行时使用 1 个模板 .

如果您将 View 更改为呈现文本框而不是编辑器,它会起作用,我只是尝试过:

<%@ Control Language="C#" Inherits="System.Web.Mvc.ViewUserControl" %>
<% var modelMetadata = ViewData.ModelMetadata; %>
<% if (modelMetadata.HideSurroundingHtml)
{ %>
<%= Html.Editor(modelMetadata.PropertyName) %>
<% }
else
{ %>
<% if (!String.IsNullOrEmpty(modelMetadata.DisplayName))
{ %>
<div class="editor-label"><%= Html.Label(modelMetadata.PropertyName) %></div>
<% } %>
<div class="editor-field"><%= Html.TextBox(modelMetadata.PropertyName) %> <%= Html.ValidationMessage(modelMetadata.PropertyName) %></div>
<% } %>

如果您想呈现其他内容而不是文本框(即下拉列表),您需要在模板中为该属性确定并呈现它。或者,如果您有更多编辑器的共同点,我通常将其提取到共享文件夹中的部分 View 中,然后使用 Html.Partial("ViewName")
并且,关于标签的渲染与 DisplayName 无关,为了防止标签在没有显示名称的情况下渲染,请将 if 条件更改为 !String.IsNullOrEmpty(modelMetadata.DisplayName) (我已经把它放在主代码块中了)

编辑
此编辑是指与 object.ascx 默认编辑器模板相关的问题。
这是object.ascx 的代码,取自 Brad Wilson's blog :
<%@ Control Language="C#" Inherits="System.Web.Mvc.ViewUserControl" %>
<% if (ViewData.TemplateInfo.TemplateDepth > 1) { %>
<%= ViewData.ModelMetadata.SimpleDisplayText%>
<% }
else { %>
<% foreach (var prop in ViewData.ModelMetadata.Properties.Where(pm => pm.ShowForEdit
&& !ViewData.TemplateInfo.Visited(pm))) { %>
<% if (prop.HideSurroundingHtml) { %>
<%= Html.Editor(prop.PropertyName) %>
<% }
else { %>
<% if (!String.IsNullOrEmpty(Html.Label(prop.PropertyName).ToHtmlString())) { %>
<div class="editor-label"><%= Html.Label(prop.PropertyName) %></div>
<% } %>
<div class="editor-field">
<%= Html.Editor(prop.PropertyName) %>
<%= Html.ValidationMessage(prop.PropertyName, "*") %>
</div>
<% } %>
<% } %>
<% } %>

这段代码确实从编辑器内部调用了 Html.Editor,但是在一个循环中,它为复杂模型的属性创建了编辑器列表。这些调用中的每一个都将调用相应的编辑器(即,对于字符串,它将在那里显示 string.ascx 等),并且仅当您有一些不是字符串的“未知”属性并且没有特定的编辑器时(即字节 [])它会为它调用 object.ascx,但这是 不是 为当前属性调用编辑器(您正在尝试做什么):

对象模板的主要职责是显示复杂对象的所有属性,以及每个属性的标签。但是,它还负责显示模型的 NullDisplayText 的值(如果它为 null),并且还负责确保您只显示一级属性(也称为对象的“浅潜水”)。在下一篇博文中,我们将讨论自定义此模板的方法,包括执行“深潜”操作。

总结

简短版本:

相同属性的更多编辑器基本上是功能差异的解决方案(“对于是/否,我想要这里的广播组和那里的下拉列表)”,对于视觉差异,应该使用部分 View ,因为您可以根据需要嵌套它们,因为您明确地按名称调用它们,因此没有强加限制,您有责任防止任何潜在的递归。

长版:

我一直在调查这个,因为我有同样的问题,我正在使用编辑器模板来呈现 <li><td>元素(取决于配置/主题)并从它内部调用另一个包含标签和输入的编辑器(两种情况相同,但如果属性为 bool 则输入在标签之前),我再次调用第三个模板进行输入(以防止为标签/输入和输入/标签场景复制代码),但这不起作用。虽然我没有在 msdn 或其他相关来源上找到解释,但我发现编辑器什么都不给的唯一情况是当你想为当前编辑器的上下文的属性呈现编辑器时(所以它实际上正是我已经引用:“对于给定的类型,在给定的 View 特定上下文中,您只能在运行时使用 1 个模板。”)。在对此进行了更多思考之后,我现在相信他们施加此限制是正确的,因为属性 x 只能使用一个编辑器呈现。您可以根据需要为属性 x 设置任意数量的编辑器,但是您不能使用多个模板渲染一次属性。用于渲染属性 x 的任何模板都可以使用其他模板来渲染属性 x 的 PARTS,但您不能对 x 多次使用(相同或不同)编辑器(相同的逻辑适用于具有两个或多个属性 x(相同类型)和名称)在同一型号上)。

此外,如果您可以将当前属性的另一个模板插入当前模板,则可以为当前属性链接任意数量的模板,并且很容易导致递归,因此以一种或另一种方式会导致您出现计算器溢出:)

关于.net - 用于属性的 ASP.NET MVC 编辑器模板,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12192225/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com