<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 { get; set; }
[MaxLength(128)]
[ColumnLayout(DetailWidthsAsString = "100 50")]
[Required]
[Display(Name = "Name", Order = 400)]
public string Name { get; set; }
[ColumnLayout(DetailWidthsAsString = "60 30")]
[Display(Name = "Price", Order = 300)]
public decimal Price { get; set; }
[Display(Name = "Cur", Order = 280)]
[ColumnLayout(WidthsAsString = "10", DetailWidthsAsString = "40 20")]
public Currency ChosenCurrency { get; set; }
[ColumnLayout(DetailWidthsAsString = "20")]
[Display(Name = "Av", Order = 230)]
public bool Available { get; set; }
}
public class SimpleProductViewModelDetail: SimpleProductViewModel
{
[MaxLength(256)]
[Display(Name = "Description", Order = 100)]
[ColumnLayout(DetailWidthsAsString = "80")]
[DisplayFormat(NullDisplayText = "no description available")]
public string Description { get; set; }
}
public class SimpleProductlistViewModel
{
public DataPage<SimpleProductViewModel> Products { get; set; }
}
public class SimpleProductlistBatchViewModel: SimpleProductlistViewModel
{
public IEnumerable<SimpleProductViewModel> ModifiedProducts { get; set; }
}
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 .pagination, thead .pagination
{
margin: 0;
display: inline !important;
}
.full-cell {
width: 100%;
}
table .input-validation-error {
border: 1px solid red;
background-color: mistyrose;
}