[英]Blazor validation for custom class
我正在測試 Blazor,但遇到了驗證問題。 在驗證一個簡單的類時,我可以只使用注釋。 如果我有自己的自定義類,盡管驗證不會對我的自定義類中的所有內容運行。 該問題似乎特定於 Blazor,因為我可以在 ASP 中使用此驗證。
這是我的兩個簡單模型:
public class TestModel
{
[Required]
[Range(12, 400, ErrorMessage = "This works")]
public int Count { get; set; }
public KeyValue KeyValues { get; set; }
public TestModel()
{
Count = 4;
KeyValues = new KeyValue()
{
Key = 5,
Value = "str"
};
}
}
和 KeyValue 類
public class KeyValue
{
[Required]
[Range(10, 300, ErrorMessage = "This number check doesn't")]
public int Key { get; set; }
[Required]
[StringLength(10, MinimumLength = 5, ErrorMessage = "Nor the string one")]
public string Value { get; set; }
}
這就是我的組成部分。 它驗證 Model.Count 屬性,但不驗證嵌套類。
<EditForm Model="@Model" OnValidSubmit="@DoStuff">
<DataAnnotationsValidator />
<ValidationSummary />
<div class="row">
<div class="col-md-4">
<input type="number" bind="@Model.Count" class="form-control" placeholder="Condition property name" />
</div>
<div class="col-md-4">
<input type="number" bind="@Model.KeyValues.Key" class="form-control" placeholder="Condition property name" />
</div>
<div class="col-md-4">
<InputText bind-Value="@Model.KeyValues.Value"></InputText>
</div>
</div>
<div class="row">
<div class="col-md-12">
<button type="submit" class="btn btn-info">Create</button>
</div>
</div>
</EditForm>
這是 Blazor 的一個已知限制,但您可以解決它。
首先,在<EditForm>
上使用OnSubmit
事件而不是OnValidSubmit
。 該方法被傳遞一個EditContext
像這樣......
private void FormSubmitted(EditContext context)
{
...
}
如果您使用以下擴展,您可以在FormSubmitted
方法中使用以下代碼,它不僅會驗證您的整個對象樹,還會根據結果更新您的 UI。
{
if (context.ValdiateObjectTree())
{
... do whatever
}
}
擴展...
using Microsoft.AspNetCore.Components.Forms;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
namespace PeterLeslieMorris.Blazor.Validation.Extensions
{
public static class EditContextExtensions
{
static PropertyInfo IsModifiedProperty;
static MethodInfo GetFieldStateMethod;
/// <summary>
/// Validates an entire object tree
/// </summary>
/// <param name="editContext">The EditContext to validate the Model of</param>
/// <returns>True if valid, otherwise false</returns>
public static bool ValidateObjectTree(this EditContext editContext)
{
var validatedObjects = new HashSet<object>();
ValidateObject(editContext, editContext.Model, validatedObjects);
editContext.NotifyValidationStateChanged();
return !editContext.GetValidationMessages().Any();
}
public static void ValidateProperty(this EditContext editContext, FieldIdentifier fieldIdentifier)
{
if (fieldIdentifier.Model == null)
return;
var propertyInfo = fieldIdentifier.Model.GetType().GetProperty(
fieldIdentifier.FieldName,
BindingFlags.Public | BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Static);
var validatedObjects = new HashSet<object>();
ValidateProperty(editContext, fieldIdentifier.Model, propertyInfo, validatedObjects);
}
private static void ValidateObject(
EditContext editContext,
object instance,
HashSet<object> validatedObjects)
{
if (instance == null)
return;
if (validatedObjects.Contains(instance))
return;
if (instance is IEnumerable && !(instance is string))
{
foreach (object value in (IEnumerable)instance)
ValidateObject(editContext, value, validatedObjects);
return;
}
if (instance.GetType().Assembly == typeof(string).Assembly)
return;
validatedObjects.Add(instance);
var properties = instance.GetType().GetProperties(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
foreach (PropertyInfo property in properties)
ValidateProperty(editContext, instance, property, validatedObjects);
}
private static void ValidateProperty(
EditContext editContext,
object instance,
PropertyInfo property,
HashSet<object> validatedObjects)
{
NotifyPropertyChanged(editContext, instance, property.Name);
object value = property.GetValue(instance);
ValidateObject(editContext, value, validatedObjects);
}
private static void NotifyPropertyChanged(
EditContext editContext,
object instance,
string propertyName)
{
if (GetFieldStateMethod == null)
{
GetFieldStateMethod = editContext.GetType().GetMethod(
"GetFieldState",
BindingFlags.NonPublic | BindingFlags.Instance);
}
var fieldIdentifier = new FieldIdentifier(instance, propertyName);
object fieldState = GetFieldStateMethod.Invoke(editContext, new object[] { fieldIdentifier, true });
if (IsModifiedProperty == null)
{
IsModifiedProperty = fieldState.GetType().GetProperty(
"IsModified",
BindingFlags.Public | BindingFlags.Instance);
}
object originalIsModified = IsModifiedProperty.GetValue(fieldState);
editContext.NotifyFieldChanged(fieldIdentifier);
IsModifiedProperty.SetValue(fieldState, originalIsModified);
}
}
}
您可以在此處找到擴展源。 您也可以使用Blazor-Validation ,它也允許您使用FluentValidation 。
如果您想更深入地了解 Blazor 表單/驗證的工作原理,可以在Blazor 大學的這一部分閱讀相關內容。
如果其他人偶然發現這個問題,現在不可能。 根據這篇文章,它應該在 3.0.0-preview8 中可用https://github.com/aspnet/AspNetCore/issues/10896
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.