Ненавязчивое пользовательское правило MVC работает только в одной форме на клиенте
Я создал собственное правило проверки для одного конкретного поля в своем приложении MVC 5. Он отлично работает в форме редактирования, но при проверке того же поля в форме "создать" проверка на стороне клиента не запускается - проверка на стороне клиента запускается, но в форме создания она отображается как действительная, хотя я могу Видите, это не так. Так что сообщение не отображается.
Обе формы используют одну и ту же модель. Сценарии добавляются на страницу _layout, поэтому оба вида имеют все сценарии. Оба представления имеют одинаковый код бритвы, включая ValidationMessageFor()
Когда форма попадает в контроллер, модель недействительна из-за пользовательской ошибки. Таким образом, проверка работает на стороне сервера, а не на клиенте. Я не могу найти ничего, что могло бы заставить его работать в одной форме, но не в другой.
Вот мой код:
Пользовательский атрибут:
public class AtLeastOneRequiredAttribute : ValidationAttribute, IClientValidatable
{
public string OtherPropertyNames;
public AtLeastOneRequiredAttribute(string otherPropertyNames)
{
OtherPropertyNames = otherPropertyNames;
}
protected override ValidationResult IsValid(object value, ValidationContext validationContext)
{
string[] propertyNames = OtherPropertyNames.Split(',');
bool IsAllNull = true;
foreach(var i in propertyNames)
{
var p = validationContext.ObjectType.GetProperty(i);
var val = p.GetValue(validationContext.ObjectInstance, null);
if(val != null && val.ToString().Trim() != "")
{
IsAllNull = false;
break;
}
}
if(IsAllNull)
{
return new ValidationResult(FormatErrorMessage(validationContext.DisplayName));
}
else
{
return null;
}
}
public IEnumerable<ModelClientValidationRule> GetClientValidationRules(ModelMetadata metadata, ControllerContext context)
{
var rules = new ModelClientValidationRule()
{
ErrorMessage = FormatErrorMessage(metadata.DisplayName),
ValidationType = "atleastonerequired"
};
rules.ValidationParameters["otherpropertynames"] = OtherPropertyNames;
yield return rules;
}
}
Код клиента:
$(function() {
$.validator.unobtrusive.adapters.addSingleVal("atleastonerequired", "otherpropertynames");
$.validator.addMethod("atleastonerequired", function (value, element, params) {
var param = params.toString().split(',');
var IsAllNull = true;
$.each(param, function (i, val) {
var valueOfItem = $('#Activity_' + val).val().trim();
if (valueOfItem != '') {
IsAllNull = false;
return false;
}
});
if (IsAllNull) {
return false;
}
else {
return true;
}
})
})
Вид - Редактировать и Создать Формы идентичны:
@using (Html.BeginForm("Edit", "Activity", FormMethod.Post, new { @id = "editActivityForm" }))
{
@Html.AntiForgeryToken()
<div class="form activity-form">
<hr />
@Html.ValidationSummary(true, "", new { @class = "text-danger" })
@Html.HiddenFor(model => model.Activity.RecordId)
<div class="form-group">
@Html.LabelFor(model => model.Activity.Acres, htmlAttributes: new { @class = "control-label" })
@Html.EditorFor(model => model.Activity.Acres, new { htmlAttributes = new { @class = "form-control" } })
@Html.ValidationMessageFor(model => model.Activity.Acres, "", new { @class = "text-danger" })
</div>
<div class="form-group">
@Html.LabelFor(model => model.Activity.Volume, htmlAttributes: new { @class = "control-label" })
@Html.EditorFor(model => model.Activity.Volume, new { htmlAttributes = new { @class = "form-control" } })
@Html.ValidationMessageFor(model => model.Activity.Volume, "", new { @class = "text-danger" })
</div>
<div class="form-group">
@Html.LabelFor(model => model.Activity.Feet, htmlAttributes: new { @class = "control-label" })
@Html.EditorFor(model => model.Activity.Feet, new { htmlAttributes = new { @class = "form-control" } })
@Html.ValidationMessageFor(model => model.Activity.Feet, "", new { @class = "text-danger" })
</div>
<div class="form-group">
@Html.LabelFor(model => model.Activity.Hours, htmlAttributes: new { @class = "control-label" })
@Html.EditorFor(model => model.Activity.Hours, new { htmlAttributes = new { @class = "form-control" } })
@Html.ValidationMessageFor(model => model.Activity.Hours, "", new { @class = "text-danger" })
</div>
<div class="form-group">
@Html.LabelFor(model => model.Activity.Comment, htmlAttributes: new { @class = "control-label" })
@Html.EditorFor(model => model.Activity.Comment, new { htmlAttributes = new { @class = "form-control" } })
@Html.ValidationMessageFor(model => model.Activity.Comment, "", new { @class = "text-danger" })
</div>
<div class="form-group">
<input type="submit" value="Save" class="btn btn-primary" onclick="$.validator.unobtrusive.parse($('#editActivityForm'));" />
</div>
</div>
}
Модель с добавленным атрибутом:
[AtLeastOneRequired("Acres,Volume,Feet,Hours", ErrorMessage = "Activity requires at least one measure - Acres, Volume, Feet or Hours.")]
public Nullable<int> Acres { get; set; }
public Nullable<int> Volume { get; set; }
public Nullable<int> Feet { get; set; }
public Nullable<int> Hours { get; set; }
1 ответ
Проблема была с клиентским кодом. Я наконец-то обнаружил, что код клиента не ударил. В конце концов я обнаружил, что это произошло потому, что добавление проверки было внутри (function() {}). Я удалил этот "готовый" кусок, и теперь он работает каждый раз.