3

C#在 ASP.NET MVC 5 框架的顶部编写了一个项目。我试图将我的视图与我的视图模型分离,以便我可以使我的视图可重用。通过大量使用,EditorTemplates我能够通过评估ModelMetadata模型上每个属性的数据注释属性和数据注释属性来创建所有标准视图(即创建、编辑和详细信息),然后渲染页面。我唯一感到困惑的是如何渲染Index视图。

我的索引视图通常接受一个IEnumerable<object>IPagedList<object>集合。在我看来,我希望能够评估集合中每个object/record 的 ModelMetadata 以确定是否object应该显示上的属性。

换句话说,我的视图模型看起来像这样

public class DisplayPersonViewModel
{
    public int Id{ get; set; }

    [ShowOnIndexView]
    public string FirstName { get; set; }

    [ShowOnIndexView]
    public string LastName { get; set; }

    [ShowOnIndexView]
    public int? Age { get; set; }

    public string Gender { get; set; }
}

然后我的Index.cshtml视图将接受IPagedList<DisplayPersonViewModel>集合中的每条记录,我想显示用属性修饰的ShowOnIndexView属性的值。

通常,我可以用类似这样的方法在我的视图中评估 ModelMetadata

@model IPagedList<object>
@{
var elements = ViewData.ModelMetadata.Properties.Where(metadata => !metadata.IsComplexType && !ViewData.TemplateInfo.Visited(metadata))
                                                .OrderBy(x => x.Order)
                                                .ToList();
}

<tr>
    @foreach(var element in elements)
    {
        var onIndex = element.ContainerType.GetProperty(element.PropertyName)
                             .GetCustomAttributes(typeof(ShowOnIndexView), true)
                             .Select(x => x as ShowOnIndexView)
                             .FirstOrDefault(x => x != null);
        if(onIndex == null) 
        {
            continue;
        }

        @Html.Editor(element.PropertyName, "ReadOnly")
    }
</tr>

然后我的控制器看起来像这样

public class PersonController : Controller
{
        public ActionResult Index()
        {
            // This would be a call to a service to give me a collection with items. but I am but showing the I would pass a collection to my view
            var viewModel = new List<DisplayPersonViewModel>();

            return View(viewModel);
        }
}

然而,评估的问题ModelMetadata在于IPagedList<DisplayPersonViewModel>它给了我关于集合本身的信息,而不是关于集合中的泛型类型或单个模型的信息。换句话说,我得到诸如总页数、每页项目数、总记录数之类的信息。

问题

如何访问ModelMetadata集合中每一行的信息,以便能够知道要显示哪些属性,哪些不显示?

4

1 回答 1

2

我将通过建议您不要用这种类型的代码污染您的视图来作为这个答案的开头。更好的解决方案是创建一个自定义HtmlHelper扩展方法来生成 html,这为您提供了更大的灵活性,可以进行单元测试,并且可以重用。

您需要更改的第一件事是视图中的模型声明,它需要

@model object

否则你会抛出这个异常List<DisplayPersonViewModel>不是IEnumerable<object>IPagedList<object>,但它是object

请注意,尚不清楚您是想要集合中的ModelMetadatafortype还是集合中的每个项目,所以我将两者都包含在内,以及获取type

@model object
@{
    var elements = ViewData.ModelMetadata.Properties.Where(metadata => !metadata.IsComplexType && !ViewData.TemplateInfo.Visited(metadata)).OrderBy(x => x.Order).ToList();

    // Get the metadata for the model
    var collectionMetaData = ViewData.ModelMetadata;
    // Get the collection type
    Type type = collectionMetaData.Model.GetType();
    // Validate its a collection and get the type in the collection
    if (type.IsGenericType)
    {
        type = type.GetInterfaces().Where(t => t.IsGenericType)
            .Where(t => t.GetGenericTypeDefinition() == typeof(IEnumerable<>))
            .Single().GetGenericArguments()[0];
    }
    else if (type.IsArray)
    {
        type = type.GetElementType();
    }
    else
    {
        // its not a valid model
    }
    // Get the metadata for the type in the collection
    ModelMetadata typeMetadata = ModelMetadataProviders.Current.GetMetadataForType(null, type);
}
....
@foreach (var element in collectionMetaData.Model as IEnumerable))
{
    // Get the metadata for the element
    ModelMetadata elementMetadata = ModelMetadataProviders.Current.GetMetadataForType(() => element, type);
    ....

请注意,使用反射来确定循环的每次迭代中是否存在属性是低效的,我建议您这样做一次(基于typeMetadata)。然后,您可以(例如)生成一个bool值数组,然后在循环中使用索引器来检查值)。更好的选择是让您的ShowOnIndexView属性实现IMetadataAware并向 of 添加一个值AdditionalValuesModelMetadata以便var onIndex根本不需要代码。有关实现的示例IMetadataAware,请参阅CustomAttribute 反映 html 属性 MVC5

于 2018-08-04T08:34:47.493 回答