Batch server grid

Name Price Cur Av
<form asp-antiforgery="true">
    <div asp-validation-summary="All" class="text-danger"></div>
    <grid asp-for="ModifiedProducts"
          type="Batch"
          all-properties="true"
          mvc-controller="typeof(LiveExamples.Controllers.GridsController)"
          row-id="batch-example"
          operations="user => Functionalities.FullInLine"
          class="table table-condensed table-bordered">
        <toolbar zone-name="@LayoutStandardPlaces.Header">
            <pager mode="CustomPages"
                   class="pagination pagination-sm"
                   max-pages="4"
                   current-page="Products.Page"
                   page-size-default="5"
                   total-pages="Products.TotalPages"
                   skip-url-token="_zzz_"
                   url-default="@Url.Action("BatchServer",
                               "Grids"new {page="_zzz_" })" />
            <button type="button" data-operation="add append 0"
                    class="btn btn-sm btn-primary">
                new product
            </button>
        </toolbar>
    </grid>
    @*Compulsory! Store original values and data-page information*@
    <store-model asp-for="Products" encrypted="true" />
    <div class="form-group col-xs-12">
        <button type="submit" class="btn btn-primary">
            Submit changes
        </button>
        <button type="reset" class="btn btn-primary">
            Reset
        </button>
    </div>
</form>
/*
 DetailWidthsAsString in ColumnLayout specifies percentage width 
 for different device screen widths, while WidthsAsString
 percentage width when shown in-line within a collection control.
 Oder specify the order columns are rendered. Higher Order come first.
 All setting, included Display/Name may be overriden by providing
 a column definition TagHelper within the control definition.
*/
public class SimpleProductViewModel 
{
    public int? Id { getset; }
    [MaxLength(128)]
    [ColumnLayout(DetailWidthsAsString = "100 50")]
    [Required]
    [Display(Name = "Name", Order = 400)]
    public string Name { getset; }
    [ColumnLayout(DetailWidthsAsString = "60 30")]
    [Display(Name = "Price", Order = 300)]
    public decimal Price { getset; }
    [Display(Name = "Cur", Order = 280)]
    [ColumnLayout(WidthsAsString = "10", DetailWidthsAsString = "40 20")]
    public Currency ChosenCurrency { getset; }
    [ColumnLayout(DetailWidthsAsString = "20")]
    [Display(Name = "Av", Order = 230)]
    public bool Available { getset; }
 
}   
 
public class SimpleProductViewModelDetailSimpleProductViewModel
{
    [MaxLength(256)]
    [Display(Name = "Description", Order = 100)]
    [ColumnLayout(DetailWidthsAsString = "80")]
    [DisplayFormat(NullDisplayText = "no description available")]
    public string Description { getset; }
}
public class SimpleProductlistViewModel
{
    public DataPage<SimpleProductViewModel> Products { getset; }
}
public class SimpleProductlistBatchViewModelSimpleProductlistViewModel
{
    public IEnumerable<SimpleProductViewModel> ModifiedProducts { getset; }
}
public async Task<IActionResult> BatchServer(int? page)
{
    int pg = page.HasValue ? page.Value : 1;
    if (pg < 1) pg = 1;
 
    var model = new SimpleProductlistBatchViewModel
    {
        Products = await Repository.GetPage<SimpleProductViewModel>(
        null,
        q => q.OrderBy(m => m.Name),
        pg, 5)
    };
    model.ModifiedProducts = model.Products.Data;
    return View(model);
}
[HttpPost]
public async Task<IActionResult> BatchServer(SimpleProductlistBatchViewModel model)
{
    if (ModelState.IsValid)
    {
        Repository.UpdateList(false, model.Products.Data, model.ModifiedProducts);
        await Repository.SaveChanges();
        return RedirectToAction("BatchServer"new { page = model.Products.Page });
    }
    else
    {
        return View(model);
    }
}
tfoot .paginationthead .pagination
 {
    margin0;
    displayinline !important;
}
 
.full-cell {
    width100%;
}
 
table .input-validation-error {
    border1px solid red;
    background-colormistyrose;
}

Fork me on GitHub