我正在为某个viewModel属性开发客户端和服务器端验证。
在我提出的.cshtml
文件中:
@Html.DropDownListFor(model => model.EntityType.ParentId, Model.ParentTypeList, "")
@Html.ValidationMessageFor(model => model.EntityType.ParentId)
在用于业务验证的Controller中
catch (BusinessException e)
{
ModelState.AddModelError("EntityType.ParentId", Messages.CircularReference);
}
以上情况按预期工作:如果捕获到异常,则会在下拉列表旁边显示该消息。
然而,我发现这种方式并不是很优雅。在cshtml
中,我使用一种方法生成有关验证的所有必需信息。在控制器中,我必须知道确切的Key字符串并使用它。
有没有更好的方法呢?
答案 0 :(得分:23)
您可以编写一个扩展方法,该方法将为键而不是字符串采用lambda表达式:
public static class ModelStateExtensions
{
public static void AddModelError<TModel, TProperty>(
this ModelStateDictionary modelState,
Expression<Func<TModel, TProperty>> ex,
string message
)
{
var key = ExpressionHelper.GetExpressionText(ex);
modelState.AddModelError(key, message);
}
}
然后使用此方法:
catch (BusinessException e)
{
ModelState.AddModelError<MyViewModel, int>(
x => x.EntityType.ParentId,
Messages.CircularReference
);
}
答案 1 :(得分:4)
我遵循@Darin Dimitrov解决方案,但我想避免<MyViewModel, int>
因此我使用了一些不同的方法,但为此您需要MyViewModel object variable.
public static class ModelStateExtensions
{
public static void AddModelError<TModel, TProperty>(this TModel source,
Expression<Func<TModel, TProperty>> ex,
string message,
ModelStateDictionary modelState)
{
var key = System.Web.Mvc.ExpressionHelper.GetExpressionText(ex);
modelState.AddModelError(key, message);
}
}
使用方法:
catch (BusinessException e)
{
objMyViewModel.AddModelError(x => x.EntityType.ParentId,
Messages.CircularReference,
ModelState);
}
答案 2 :(得分:-1)
您希望验证在客户端和服务器端进行,并且您还在寻找一个优雅的解决方案,然后为什么可以尝试创建自定义ValidationAttribute
。