23

我有一个大模型(大我的意思是模型类包含很多字段/属性,每个都至少有一个验证属性(例如Required,MaxLengthMinLength))。我不想创建一个包含大量输入的视图来让用户用数据填充模型,我想创建几个视图,用户将在其中填充部分模型字段并进入下一步(某种“向导”)。在步骤之间重定向时,我将未填充的模型对象存储在Session. 如下所示:

模型:

public class ModelClass
{
    [MaxLength(100)] ...
    public string Prop1{get;set;}
    [MaxLength(100)] ...
    public string Prop2{get;set;}
    ...
    [Required][MaxLength(100)] ...
    public string Prop20{get;set;}
}

控制器:

[HttpPost]
public ActionResult Step1(ModelClass postedModel)
{    
    // user posts only for example Prop1 and Prop2
    // so while submit I have completly emty model object
    // but with filled Prop1 and Prop2
    // I pass those two values to Session["model"]
    var originalModel = Session["model"] as ModelClass ?? new ModelClass();
    originalModel.Prop1 = postedModel.Prop1;
    originalModel.Prop2 = postedModel.Prop2;
    Session["model"] = originalModel;

    // and return next step view
    return View("Step2");
}

[HttpPost]
public ActionResult Step2(ModelClass postedModel)
{
    // Analogically the same
    // I have posted only Prop3 and Prop4

    var originalModel = Session["model"] as ModelClass;
    if (originalModel!=null)
    {
        originalModel.Prop3 = postedModel.Prop3;
        originalModel.Prop4 = postedModel.Prop4;
        Session["model"] = originalModel;

        // return next step view
        return View("Step3");
    }
    return View("SomeErrorViewIfSessionBrokesSomeHow")
}

Step1视图只有Prop1and的输入Prop2,Step2 视图包含 and 的输入Prop3Prop4

但事情是这样的

例如,当用户打开时,步骤 1 并用长度超过 100 个字符的值填充 Prop1 客户端验证工作正常。但是,当然,我必须在控制器的服务器端验证这个值。如果我有完整的模型,我只需执行以下操作:

if(!ModelState.IsValid) return View("the same view with the same model object");

所以用户必须再次填写表格并更正。 但是在第 1 步用户只填写了 20 个属性,我需要验证它们。我不能使用ModelState.IsValid,因为模型状态将无效。如您所见Prop20,标记为[Required]属性,当用户提交时Prop1Prop2Prop20空,这ModelState就是无效的原因。当然,我可以允许用户进入第 2 步,填写所有步骤并仅在最后一步验证模型状态,但我不想让用户进入第 2 步,如果他填写的第 1 步不正确。我希望在控制器中进行此验证。所以问题是: 我怎样才能只验证模型的一部分?如何验证只有部分模型属性与它们的验证属性匹配?

4

4 回答 4

16

一种可能的解决方案:

  1. 使用 ModelState.IsValidField(string key);

    if (ModelState.IsValidField("Name") && ModelState.IsValidField("Address"))
    { ... }
    

然后在一切完成后使用:

if(ModelState.IsValid) { .. }
于 2013-01-11T15:32:55.063 回答
11

我认为最优雅的方法是这样做:

List<string> PropertyNames = new List<string>()
{
    "Prop1",
    "Prop2"
};

if (PropertyNames.Any(p => !ModelState.IsValidField(p)))
{
    // Error
}
else
{
    // Everything is okay
}

或者:

List<string> PropertyNames = new List<string>()
{
    "Prop1",
    "Prop2"
};

if (PropertyNames.All(p => ModelState.IsValidField(p)))
{
    // Everything is okay
}
else
{
    // Error
}
于 2013-01-11T15:44:47.273 回答
6

在 MVC Core 中,这将相当于:

if (ModelState.GetFieldValidationState("Name") == Microsoft.AspNetCore.Mvc.ModelBinding.ModelValidationState.Valid)
{
    // do something
}

但是,我建议在这种情况下简单地创建一个单独的视图模型。

您的局部视图模型可以由您的较大视图模型继承,因此您不必在代码中重复自己(DRY 主体)。

最好避免对属性名称进行硬编码!

于 2018-01-08T14:40:19.180 回答
5

只是为了添加到现有的答案。我不会硬编码属性名称,而是使用一个属性与其余的验证属性一起添加,如下所示:

public class ValidationStageAttribute : Attribute
{
    public int StageNumber { get; private set; }
    public ValidationStageAttribute(int stageNumber)
    {
        StageNumber = stageNumber;
    }
}

现在我们可以在不了解模型本身的情况下获取属性名称,可以将部分验证提取到方法中(如果您经常使用它,您的基本控制器将是一个好地方)。

protected bool ValidateStage(object viewModel, int stageToValidate)
{
    var propertiesForStage = viewModel.GetType()
        .GetProperties()
        .Where(prop => prop.GetCustomAttributes(false).OfType<ValidationStageAttribute>().Any(attr => attr.StageNumber == stageToValidate))
        .Select(prop => prop.Name);

    return propertiesForStage.All(p => ModelState.IsValidField(p));
}

现在,您在发布操作中需要做的就是致电ValidateStage(viewModel, 1)

于 2016-07-20T13:57:34.337 回答