5

我有两个类用作两个不同视图的模型。您可以看到第二个类包含第一个类的实例。第一个包含远程验证属性。

[MetadataType( typeof( ExceptionLogModel.EmailRecipientMetadata ) )]
public class EmailRecipientViewModel
{
    public int EmailRecipientID { get; set; }

    [Remote( "ValidateEmailRecipientNameUniqueness", "EmailRecipient", ErrorMessage = "Name is not unique." )]
    public string Name { get; set; }

    [Remote( "ValidateEmailRecipientEmailUniqueness", "EmailRecipient", ErrorMessage = "Email is not unique." )]
    public string Email { get; set; }
}

public class EmailRecipientChoices
{
    public List<EmailRecipient> UnselectedEmailRecipients { get; set; }
    public List<EmailRecipient> SelectedEmailRecipients { get; set; }
    public EmailRecipientViewModel EmailRecipient { get; set; }
}

当这些验证在浏览器中触发时,会根据视图使用的类发出两个不同的请求。可以看到查询字符串参数名称不同:

http://localhost:55327/EmailRecipient/ValidateEmailRecipientNameUniqueness?Name=sdhsdgh

http://localhost:55327/EmailRecipient/ValidateEmailRecipientNameUniqueness?EmailRecipient.Name=sdhsdgh

这是我的操作方法的当前版本,它不适用于第二个 URL:

public JsonResult ValidateEmailRecipientNameUniqueness( string name )
{
    var isValid = !_emailRecipientRepo.NameExists( name );

    return Json( isValid, JsonRequestBehavior.AllowGet );
}

当使用第二个 URL 时,name 参数将为空。我读过我应该能够向该参数添加一个 Bind 属性并添加一个前缀,但这也不起作用。我什至尝试将前缀设置为EmailRecipient.以防万一它需要点。为了以防万一,我还尝试在名称中使用大写 N。不去。添加它也会破坏其他 URL!

public JsonResult ValidateEmailRecipientNameUniqueness( [Bind( Prefix = "EmailRecipient")] string name )

可能的解决方案

我可以让该方法获取一个 EmailRecipientViewModel 的实例并为其创建一个 IModelBinder,我可以在其中查找任一命名约定并将其分配给该实例。这似乎比应该做的工作更多。

我可以使用重载@Html.EditorFor()并告诉它为 htmlFieldName 使用“名称”,也可以使用@Html.ValidationMessage( "Name" )而不是ValidationMessageFor. 唯一的缺点是潜在的命名冲突,但这并不是什么大不了的事。我只需要为正在使用的这个类的所有实例使用一个唯一的名称。更新:实际上,如果我这样做,因为我更改了名称,所以在我发布表格时会破坏事情。那不好。

...

我刚刚发现我可以让该方法不带参数,并手动访问查询字符串。这是一个非常简单的解决方案,但我没有得到 nice 参数。

string name = Request.QueryString[ "Name" ] ?? Request.QueryString[ "EmailRecipient.Name" ];

这很容易,我可能只会使用它。但是,既然我已经输入了这个问题,我会问,有没有更优雅的解决方案?

4

2 回答 2

3

好吧,我知道这已经很晚了,但是有一个更优雅的解决方案:

public JsonResult ValidateEmailRecipientNameUniqueness (EmailRecipient recipient)
{
    string name = recipient.Name;
    var isValid = !_emailRecipientRepo.NameExists(name);
    return Json(isValid, JsonRequestBehavior.AllowGet);
}

换句话说,使用模型本身。它将正确绑定name属性,您只需要此值。

于 2015-03-05T12:43:01.110 回答
1

如果不滚动您自己的验证或模型绑定器,就没有真正干净的方法来做到这一点。把它想象成模型绑定,模型绑定器需要知道传入的名称,远程验证也是如此。您可以采取的一种方法是在您的控制器中创建两个单独的远程验证方法,最终调用一个实际完成所有验证工作的方法。

于 2011-04-14T04:45:27.320 回答