--- sidebar_position: 6 title: Creation of the form --- Open the `Pages/Add.razor` file and edit the following file: ```cshtml title="Pages/Add.razor" @page "/add"

Add

Enchant categories:

@foreach (var item in enchantCategories) { }

Repair with:

@foreach (var item in repairWith) { }

``` * The `EditForm` component is rendered where the <EditForm> appears. * The model is created in the component code and kept in a private field ( itemModel ). The field is assigned to the `Model` attribute of the <EditForm> . * The `InputText` component ( id="display-name" ) is an input component for modifying string values. The `@bind-Value` directive attribute binds the `itemModel.DisplayName` model property to the `Value` property of the `InputText` component. * The `HandleValidSubmit` method is assigned to `OnValidSubmit`. The handler is called if the form passes validation. * The Data Annotations Validator (`DataAnnotationsValidator`) attaches support for validation using Data Annotations: * If the form field <input> is not populated when the Submit button is selected, an error is displayed in the validation summary (`ValidationSummary`) ("The DisplayName field is required.") and `HandleValidSubmit` is not called. * If the form field <input> contains more than fifty characters when the submit button is selected, an error is displayed in the validation summary ("Displayed name must not exceed 50 characters.") and `HandleValidSubmit` is not called. * If the form field <input> contains a valid value when the Submit button is selected, `HandleValidSubmit` is called. ## Form code Open the `Pages/Add.razor.cs` file and edit the following file: ```csharp title="Pages/Add.razor.cs" public partial class Add { [Inject] public ILocalStorageService LocalStorage { get; set; } [Inject] public IWebHostEnvironment WebHostEnvironment { get; set; } /// /// The default enchant categories. /// private List enchantCategories = new List() { "armor", "armor_head", "armor_chest", "weapon", "digger", "breakable", "vanishable" }; /// /// The default repair with. /// private List repairWith = new List() { "oak_planks", "spruce_planks", "birch_planks", "jungle_planks", "acacia_planks", "dark_oak_planks", "crimson_planks", "warped_planks" }; /// /// The current item model /// private ItemModel itemModel = new() { EnchantCategories = new List(), RepairWith = new List() }; private async void HandleValidSubmit() { // Get the current data var currentData = await LocalStorage.GetItemAsync>("data"); // Simulate the Id itemModel.Id = currentData.Max(s => s.Id) + 1; // Add the item to the current data currentData.Add(new Item { Id = itemModel.Id, DisplayName = itemModel.DisplayName, Name = itemModel.Name, RepairWith = itemModel.RepairWith, EnchantCategories = itemModel.EnchantCategories, MaxDurability = itemModel.MaxDurability, StackSize = itemModel.StackSize, CreatedDate = DateTime.Now }); // Save the image var imagePathInfo = new DirectoryInfo($"{WebHostEnvironment.WebRootPath}/images"); // Check if the folder "images" exist if (!imagePathInfo.Exists) { imagePathInfo.Create(); } // Determine the image name var fileName = new FileInfo($"{imagePathInfo}/{itemModel.Name}.png"); // Write the file content await File.WriteAllBytesAsync(fileName.FullName, itemModel.ImageContent); // Save the data await LocalStorage.SetItemAsync("data", currentData); } private async Task LoadImage(InputFileChangeEventArgs e) { // Set the content of the image to the model using (var memoryStream = new MemoryStream()) { await e.File.OpenReadStream().CopyToAsync(memoryStream); itemModel.ImageContent = memoryStream.ToArray(); } } private void OnEnchantCategoriesChange(string item, object checkedValue) { if ((bool)checkedValue) { if (!itemModel.EnchantCategories.Contains(item)) { itemModel.EnchantCategories.Add(item); } return; } if (itemModel.EnchantCategories.Contains(item)) { itemModel.EnchantCategories.Remove(item); } } private void OnRepairWithChange(string item, object checkedValue) { if ((bool)checkedValue) { if (!itemModel.RepairWith.Contains(item)) { itemModel.RepairWith.Add(item); } return; } if (itemModel.RepairWith.Contains(item)) { itemModel.RepairWith.Remove(item); } } } ``` You can now add a new item, if you return to the list when your new item is present. ## Concept: Form and validation ### Built-in form components The Blazor framework provides built-in form components to receive and validate user input. Inputs are validated when they are changed and when a form is submitted. The available input components are listed in the following table. | Composant d’entrée | Rendu comme… | | ----------- | ----------- | | InputCheckbox | `` | | InputDate<TValue> | `` | | InputFile | `` | | InputNumber<TValue> | `` | | InputRadio<TValue> | `` | | InputRadioGroup<TValue> | Groupe d’enfants InputRadio<TValue> | | InputSelect<TValue> | `` | | InputTextArea | `