4

我有一个由许多用户控件组成的页面。此页面的视图模型相当复杂。

public class ComplexViewModel
{
    public ObjectA ObjectAProperty { get; set; }
    public List<Things> ListOfThings { get; set; }
    public List<ThingCategories> ListOfThingCategories { get; set; }
    public List<ThingTypes> ListOfThingTypes { get; set; }
    public List<ThingOptions> ListOfThingOptions { get; set; }
    public int ChosenThingCategoryId { get; set; }
    public int ChosenThingTypeId { get; set; }
    public int ChosenThingOptionId { get; set; }
    public OtherObject ObjectData { get; set; }
}

该页面还有一个 PostModel,其中包含用于过滤、排序等信息。

    public class SimplePostModel
{
    public int ChosenThingCategoryId { get; set; }
    public int ChosenThingTypeId { get; set; }
    public int ChosenThingOptionId { get; set; }
    public int ChosenThingFilterTypeId { get; set; }
    public int ChosenThingSortTypeId { get; set; }
    public int ChosenThingOtherId { get; set; }
    public int ChosenThingMoreId { get; set; }
    public int ChosenThingOMGId { get; set; }
}

验证简单的 PostModel,然后控制器打开 3 个以上的存储库,对每个存储库进行多次调用并构建视图模型。至少可以说我的控制器动作已经变得相当大了。

这是迄今为止我工作过的最复杂的页面,我很难决定如何使它更简单。

我的第一个想法是创建一个视图模型工厂,在绑定验证之后,它会调用存储库并返回 ViewModel。

然后我考虑创建一个自定义模型绑定器,该绑定器将验证 PostModel,然后一步水合 ViewModel。

所以我的问题是你如何水合一个复杂的视图模型?

在我写这篇文章的时候,我想到了使用 Html.RenderAction 并为构成这个页面野兽的每个用户控件创建一个模型。

更新:

存储库调用 WCF 服务,应用程序是更大 SOA 架构的一部分。

4

1 回答 1

6

一些一般提示。数据可以分为几类:系统范围、会话范围、请求范围。

系统范围数据是需要呈现给用户但对每个用户都相同的数据。博客应用程序的一个示例是标签云或类别列表。我认为这些数据不需要流经控制器或动作,因为它与用户交互无关。视图本身可以调用知道如何获取(最好是缓存)这些数据的 HtmlHelper(或 LayoutDataHelper)。

会话范围数据可以使用填充 ViewData.Model 上的字段的 ActionFilter 来处理。它与动作的参数没有直接关系。例如,用户名。我更喜欢表单的属性

public class SessionDataFilter : ActionFilter 
{
    public override void OnActionExecuted(ActionExecutedContext filterContext)
    {
        if (filterContext.Result is ViewResult)
        {
            var view = filterContext.Result as ViewResult;

            // hydrate session data on view.ViewData.Model
        }
    }
}

必须在操​​作中填充请求范围/特定的所有其他内容。然而,这并不意味着你必须有一种大规模的行动方法来做到这一点。我会看看你的 ViewModel 是如何组成的。正如您所建议的,如果您有需要填充的控件,则 ViewModel 中的信息很可能可以分组到相关的集合中。因此,您可能有一个 ViewModel 只组成其他较小的视图模型(“部分视图模型”)。然后,我将分解逻辑以将每个部分视图模型(和任何其他复杂逻辑)填充到其自己的可重用和隔离方法中。

处理帖子时也适用类似的抽象,尽管我会担心发布大量不相关数据的页面的可用性。您应该能够使用ActionFilters( OnActionExecuting) 来解析相关的传入数据集(并且可以选择验证)并将它们分配给操作参数。对于发布的数据,我更喜欢过滤器而不是活页夹,除非同一组数据发布到多个操作并且传入数据的形状始终相同。

祝你好运。

于 2010-01-20T06:15:36.023 回答