Grid Validation
The Telerik Grid for Blazor supports built-in validation that is enabled by default. This article describes how the Grid validation works and how to customize or disable it.
Make sure to read the Grid CRUD Operations article first.
Basics
By default, the Grid validation uses a DataAnnotationValidator
and creates an EditContext
for the row that is in add or edit mode. When you use inline or in-cell editing, the Grid renders validation messages in Validation Tooltips on hover of the invalid inputs. In popup edit mode, the Grid shows a Validation Summary in the popup.
When a row is not in edit mode, the EditContext
is null
. The Grid EditContext
is a cascading parameter, which can overrides any cascading parameters from parent components, such as an <EditForm>
that may wrap the Grid.
The built-in Grid validation is not supported with dynamic data such as ExpandoObject
, DataTable
, or Dictionary
.
The Telerik components for Blazor do not perform the actual validation of the model. Validation is managed by the
EditContext
. The role of the Telerik components is to callEditContext
methods, subscribe toEditContext
events, retrieve validation messages, and display them. If a validation scenario does not work as expected, check the behavior in a standard Blazor<EditForm>
to verify if the issue is related to the Telerik components.
Disable Validation
To disable the built-in Grid validation:
- Add
<GridSettings>
as a child tag inside the Grid. - Add a
<GridValidationSettings>
tag inside<GridSettings>
. - Set the
Enabled
parameter ofGridValidationSettings
to false.
See the example below.
Use Custom Validator
You can validate the Grid with any validator that uses the EditContext
. To change the default validator:
- Add
<GridSettings>
as a child tag inside the Grid. - Add a
<GridValidationSettings>
tag inside<GridSettings>
. - Define the custom validator in the
<ValidatorTemplate>
RenderFragment
inside<GridValidationSettings>
.
Third party validation tools are not part of Telerik UI for Blazor. Reference the required NuGet packages explicitly.
Example
The example below shows how to:
- Enable and disable validation in the Grid.
- Validate the user input with a custom validator instead of the default
DataAnnotationsValidator
.
Install the Blazored.FluentValidation
NuGet package to run the following code and refer to the FluentValidation documentation.
Use Telerik Grid for Blazor with FluentValidation
@* Requires the Blazored.FluentValidation NuGet package *@
@using Blazored.FluentValidation
@using FluentValidation
<TelerikGrid Data="@GridData"
EditMode="@GridEditMode.Inline"
OnCreate="@OnGridCreate"
OnUpdate="@OnGridUpdate">
<GridSettings>
<GridValidationSettings Enabled="@GridValidationEnabled">
<ValidatorTemplate>
<FluentValidationValidator Validator="@GridFluentValidator" />
</ValidatorTemplate>
</GridValidationSettings>
</GridSettings>
<GridToolBarTemplate>
<GridCommandButton Command="Add">Add Item</GridCommandButton>
<label class="k-checkbox-label">
<TelerikCheckBox @bind-Value="@GridValidationEnabled" />
Enable Validation
</label>
</GridToolBarTemplate>
<GridColumns>
<GridColumn Field="@nameof(Product.Name)" />
<GridColumn Field="@nameof(Product.Price)" DisplayFormat="{0:C2}" />
<GridColumn Field="@nameof(Product.Quantity)" DisplayFormat="{0:N0}" />
<GridColumn Field="@nameof(Product.ReleaseDate)" DisplayFormat="{0:d}" />
<GridColumn Field="@nameof(Product.Discontinued)" Width="120px" />
<GridCommandColumn Width="180px">
<GridCommandButton Command="Edit">Edit</GridCommandButton>
<GridCommandButton Command="Save" ShowInEdit="true">Save</GridCommandButton>
<GridCommandButton Command="Cancel" ShowInEdit="true">Cancel</GridCommandButton>
</GridCommandColumn>
</GridColumns>
</TelerikGrid>
@code {
private List<Product> GridData { get; set; } = new();
private FluentProductValidator GridFluentValidator = new();
public class FluentProductValidator : AbstractValidator<Product>
{
public FluentProductValidator()
{
RuleFor(item => item.Name).NotEmpty().MinimumLength(3).MaximumLength(24);
RuleFor(item => item.Price).NotNull().GreaterThan(0);
RuleFor(item => item.ReleaseDate).NotEmpty().GreaterThanOrEqualTo(DateTime.Today.AddYears(-10));
}
}
private bool GridValidationEnabled { get; set; } = true;
private int LastId { get; set; }
private void OnGridCreate(GridCommandEventArgs args)
{
var createdItem = (Product)args.Item;
createdItem.Id = ++LastId;
GridData.Insert(0, createdItem);
}
private void OnGridUpdate(GridCommandEventArgs args)
{
var updatedItem = (Product)args.Item;
int originalItemIndex = GridData.FindIndex(i => i.Id == updatedItem.Id);
if (originalItemIndex != -1)
{
GridData[originalItemIndex] = updatedItem;
}
}
protected override void OnInitialized()
{
for (int i = 1; i <= 5; i++)
{
GridData.Add(new Product()
{
Id = ++LastId,
Name = $"Product {LastId}",
Price = LastId % 2 == 0 ? null : Random.Shared.Next(0, 100) * 1.23m,
Quantity = LastId % 2 == 0 ? 0 : Random.Shared.Next(0, 3000),
ReleaseDate = DateTime.Today.AddDays(-Random.Shared.Next(365, 3650)),
Discontinued = LastId % 2 == 0
});
}
}
public class Product
{
public int Id { get; set; }
public string Name { get; set; } = string.Empty;
public decimal? Price { get; set; }
public int Quantity { get; set; }
public DateTime? ReleaseDate { get; set; }
public bool Discontinued { get; set; }
}
}