|  |  |  | 
|---|
|  |  |  | using Ao.Lang; | 
|---|
|  |  |  | using CMS.Extensions.Abp.AspNetCore.Mvc.Filters; | 
|---|
|  |  |  | using CMS.Plugin.HIAWms.Application.Contracts.Dtos.WmsMaterial; | 
|---|
|  |  |  | using CMS.Plugin.HIAWms.Application.Contracts.Services; | 
|---|
|  |  |  | using Microsoft.AspNetCore.Authorization; | 
|---|
|  |  |  | using Microsoft.AspNetCore.Http; | 
|---|
|  |  |  | using Microsoft.AspNetCore.Mvc; | 
|---|
|  |  |  | using MiniExcelLibs; | 
|---|
|  |  |  | using System.Reflection; | 
|---|
|  |  |  | using Volo.Abp; | 
|---|
|  |  |  | using Volo.Abp.Application.Dtos; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | namespace CMS.Plugin.HIAWms.Controller | 
|---|
|  |  |  | { | 
|---|
|  |  |  | using Ao.Lang; | 
|---|
|  |  |  | using CMS.Extensions.Abp.AspNetCore.Mvc.Filters; | 
|---|
|  |  |  | using CMS.Framework.AspNetCore.Users; | 
|---|
|  |  |  | using CMS.Plugin.HIAWms.Application.Contracts.Dtos.WmsMaterial; | 
|---|
|  |  |  | using CMS.Plugin.HIAWms.Application.Contracts.Services; | 
|---|
|  |  |  | using Microsoft.AspNetCore.Authorization; | 
|---|
|  |  |  | using Microsoft.AspNetCore.Http; | 
|---|
|  |  |  | using Microsoft.AspNetCore.Mvc; | 
|---|
|  |  |  | using MiniExcelLibs; | 
|---|
|  |  |  | using System.Reflection; | 
|---|
|  |  |  | using Volo.Abp; | 
|---|
|  |  |  | using Volo.Abp.Application.Dtos; | 
|---|
|  |  |  | using CmsQueryExtensions.Entitys; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | namespace CMS.Plugin.HIAWms.Controller | 
|---|
|  |  |  | { | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// ç©æåºç¡ä¿¡æ¯è¡¨æå¡ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | [ApiController] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSLanguageFilter))] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSUowActionFilter))] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSAuditActionFilter))] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSExceptionFilter))] | 
|---|
|  |  |  | [Route("api/v{version:apiVersion}/HIAWms/[controller]")] | 
|---|
|  |  |  | public class WmsMaterialController : ControllerBase | 
|---|
|  |  |  | { | 
|---|
|  |  |  | private readonly IWmsMaterialAppService _wmsMaterialAppService; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [ApiController] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSLanguageFilter))] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSUowActionFilter))] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSAuditActionFilter))] | 
|---|
|  |  |  | [TypeFilter(typeof(CMSExceptionFilter))] | 
|---|
|  |  |  | [Route("api/v{version:apiVersion}/HIAWms/[controller]")] | 
|---|
|  |  |  | public class WmsMaterialController : ControllerBase | 
|---|
|  |  |  | { | 
|---|
|  |  |  | private readonly IWmsMaterialAppService _wmsMaterialAppService; | 
|---|
|  |  |  | private readonly ICurrentUser _currentUser; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// Initializes a new instance of the <see cref="WmsMaterialController"/> class. | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="wmsMaterialAppService">The wmsMaterial application service.</param> | 
|---|
|  |  |  | public WmsMaterialController(IWmsMaterialAppService wmsMaterialAppService) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | _wmsMaterialAppService = wmsMaterialAppService; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public WmsMaterialController(IWmsMaterialAppService wmsMaterialAppService, ICurrentUser currentUser) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | _wmsMaterialAppService = wmsMaterialAppService; | 
|---|
|  |  |  | _currentUser = currentUser; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// è·åç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="id">æ è¯ç¬¦.</param> | 
|---|
|  |  |  | /// <param name="id">主é®ID</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | [HttpGet] | 
|---|
|  |  |  | [Route("{id}")] | 
|---|
|  |  |  | public virtual Task<WmsMaterialDto> GetAsync(Guid id) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.GetAsync(id); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [HttpGet] | 
|---|
|  |  |  | [Route("{id}")] | 
|---|
|  |  |  | public virtual Task<WmsMaterialDto> GetAsync(Guid id) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.GetAsync(id); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// å页è·åç©æåºç¡ä¿¡æ¯è¡¨çå表. | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="input">è¾å
¥.</param> | 
|---|
|  |  |  | /// <param name="input">æ¥è¯¢åæ°</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | [HttpGet] | 
|---|
|  |  |  | [Route("Page")] | 
|---|
|  |  |  | public virtual Task<PagedResultDto<WmsMaterialDto>> GetListAsync([FromQuery] GetWmsMaterialInput input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.GetListAsync(input); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [HttpGet] | 
|---|
|  |  |  | [Route("Page")] | 
|---|
|  |  |  | public virtual Task<PagedResultDto<WmsMaterialDto>> GetListAsync([FromQuery] GetWmsMaterialInput input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.GetListAsync(input); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// åå»ºç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="input">è¾å
¥.</param> | 
|---|
|  |  |  | /// <param name="input">åå»ºåæ°</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | //[Authorize] | 
|---|
|  |  |  | [HttpPost] | 
|---|
|  |  |  | public virtual Task<WmsMaterialDto> CreateAsync(WmsMaterialCreateDto input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.CreateAsync(input); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [HttpPost] | 
|---|
|  |  |  | public virtual Task<WmsMaterialDto> CreateAsync(WmsMaterialCreateDto input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | input.CreatorName = _currentUser.UserAccount;//å建人 | 
|---|
|  |  |  | return _wmsMaterialAppService.CreateAsync(input); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// æ´æ°ç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="id">æ è¯ç¬¦.</param> | 
|---|
|  |  |  | /// <param name="input">è¾å
¥.</param> | 
|---|
|  |  |  | /// <param name="id">主é®ID</param> | 
|---|
|  |  |  | /// <param name="input">æ´æ°åæ°</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | //[Authorize] | 
|---|
|  |  |  | [HttpPut] | 
|---|
|  |  |  | [Route("{id}")] | 
|---|
|  |  |  | public virtual Task<WmsMaterialDto> UpdateAsync(Guid id, WmsMaterialUpdateDto input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.UpdateAsync(id, input); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [Authorize] | 
|---|
|  |  |  | [HttpPut] | 
|---|
|  |  |  | [Route("{id}")] | 
|---|
|  |  |  | public virtual Task<WmsMaterialDto> UpdateAsync(Guid id, WmsMaterialUpdateDto input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | input.LastModifierName = _currentUser.UserAccount;//ä¿®æ¹äºº | 
|---|
|  |  |  | return _wmsMaterialAppService.UpdateAsync(id, input); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// å
éç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="ids">Idéå.</param> | 
|---|
|  |  |  | /// <param name="ids">Idéå</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | //[Authorize] | 
|---|
|  |  |  | [HttpPost] | 
|---|
|  |  |  | [Route("Clone")] | 
|---|
|  |  |  | public virtual Task<List<WmsMaterialDto>> CloneAsync([FromBody] IEnumerable<Guid> ids) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.CloneAsync(ids); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [HttpPost] | 
|---|
|  |  |  | [Route("Clone")] | 
|---|
|  |  |  | public virtual Task<List<WmsMaterialDto>> CloneAsync([FromBody] IEnumerable<Guid> ids) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | MyCurrentUser myCurrentUser = new MyCurrentUser() | 
|---|
|  |  |  | { | 
|---|
|  |  |  | UserAccount = _currentUser.UserAccount, | 
|---|
|  |  |  | UserId = _currentUser.UserId | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | return _wmsMaterialAppService.CloneAsync(ids, myCurrentUser); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// å é¤ç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="id">æ è¯ç¬¦.</param> | 
|---|
|  |  |  | /// <param name="id">主é®ID</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | //[Authorize] | 
|---|
|  |  |  | [HttpDelete] | 
|---|
|  |  |  | [Route("{id}")] | 
|---|
|  |  |  | public virtual Task DeleteAsync(Guid id) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.DeleteAsync(id); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [HttpDelete] | 
|---|
|  |  |  | [Route("{id}")] | 
|---|
|  |  |  | public virtual Task DeleteAsync(Guid id) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | MyCurrentUser myCurrentUser = new MyCurrentUser() | 
|---|
|  |  |  | { | 
|---|
|  |  |  | UserAccount = _currentUser.UserAccount, | 
|---|
|  |  |  | UserId = _currentUser.UserId | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | //return _wmsMaterialAppService.DeleteAsync(id,myCurrentUser);//é»è¾å é¤ | 
|---|
|  |  |  | return _wmsMaterialAppService.DeletePermanentlyAsync(id, myCurrentUser);//ç©çå é¤ | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// æ¹éå é¤ç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="ids">The ids.</param> | 
|---|
|  |  |  | /// <param name="ids">主é®IDéå</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | //[Authorize] | 
|---|
|  |  |  | [HttpDelete] | 
|---|
|  |  |  | public virtual Task DeleteAsync([FromBody] IEnumerable<Guid> ids) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.DeleteManyAsync(ids); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [HttpDelete] | 
|---|
|  |  |  | public virtual Task DeleteAsync([FromBody] IEnumerable<Guid> ids) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | MyCurrentUser myCurrentUser = new MyCurrentUser() | 
|---|
|  |  |  | { | 
|---|
|  |  |  | UserAccount = _currentUser.UserAccount, | 
|---|
|  |  |  | UserId = _currentUser.UserId | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | // return _wmsMaterialAppService.DeleteManyAsync(ids,myCurrentUser);//é»è¾å é¤ | 
|---|
|  |  |  | return _wmsMaterialAppService.BatchDeletePermanentlyAsync(ids, myCurrentUser);//ç©çå é¤ | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// è°æ´æåºç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="id">æ è¯ç¬¦.</param> | 
|---|
|  |  |  | /// <param name="id">主é®ID</param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | [HttpPut] | 
|---|
|  |  |  | [Route("{id}/AdjustSort/{sort}")] | 
|---|
|  |  |  | public virtual Task AdjustSortAsync(Guid id, int sort) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.AdjustSortAsync(id, sort); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [HttpPut] | 
|---|
|  |  |  | [Route("{id}/AdjustSort/{sort}")] | 
|---|
|  |  |  | public virtual Task AdjustSortAsync(Guid id, int sort) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return _wmsMaterialAppService.AdjustSortAsync(id, sort); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// å¯¼å
¥ç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | [Authorize] | 
|---|
|  |  |  | [HttpPost] | 
|---|
|  |  |  | [Route("Import")] | 
|---|
|  |  |  | public virtual async Task<IActionResult> ImportAsync(IFormFile file) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | using var stream = new MemoryStream(); | 
|---|
|  |  |  | await file.CopyToAsync(stream); | 
|---|
|  |  |  | stream.Seek(0L, SeekOrigin.Begin); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var sheetNames = stream.GetSheetNames(); | 
|---|
|  |  |  | var wmsMaterialRows = sheetNames.Contains("é
ç½®") ? MiniExcel.Query<WmsMaterialsImportModel.WmsMaterialImportModel>(stream, sheetName: "é
ç½®").ToList() : new(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!wmsMaterialRows.Any()) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | throw new UserFriendlyException("è¯·æ£æ¥å¯¼å
¥ç表格"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | await _wmsMaterialAppService.ImportAsync(new WmsMaterialsImportModel | 
|---|
|  |  |  | { | 
|---|
|  |  |  | WmsMaterials = wmsMaterialRows, | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return Ok(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | [Authorize] | 
|---|
|  |  |  | [HttpPost] | 
|---|
|  |  |  | [Route("Import")] | 
|---|
|  |  |  | public virtual async Task<IActionResult> ImportAsync(IFormFile file) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | using var stream = new MemoryStream(); | 
|---|
|  |  |  | await file.CopyToAsync(stream); | 
|---|
|  |  |  | stream.Seek(0L, SeekOrigin.Begin); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var sheetNames = stream.GetSheetNames(); | 
|---|
|  |  |  | var wmsMaterialRows = sheetNames.Contains("é
ç½®") ? MiniExcel.Query<WmsMaterialsImportModel.WmsMaterialImportModel>(stream, sheetName: "é
ç½®").ToList() : new(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!wmsMaterialRows.Any()) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | throw new UserFriendlyException("è¯·æ£æ¥å¯¼å
¥ç表格"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | MyCurrentUser myCurrentUser = new MyCurrentUser() | 
|---|
|  |  |  | { | 
|---|
|  |  |  | UserAccount = _currentUser.UserAccount, | 
|---|
|  |  |  | UserId = _currentUser.UserId | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | await _wmsMaterialAppService.ImportAsync(new WmsMaterialsImportModel | 
|---|
|  |  |  | { | 
|---|
|  |  |  | WmsMaterials = wmsMaterialRows, | 
|---|
|  |  |  | }, myCurrentUser); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return Ok(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// å¯¼åºç©æåºç¡ä¿¡æ¯è¡¨ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | [HttpGet] | 
|---|
|  |  |  | [Route("Export")] | 
|---|
|  |  |  | public virtual async Task<IActionResult> ExportAsync([FromQuery] GetWmsMaterialInput input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | input.MaxResultCount = int.MaxValue; | 
|---|
|  |  |  | var exportData = await _wmsMaterialAppService.ExportAsync(input); | 
|---|
|  |  |  | var templatePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, $"Resources/Templates/WmsMaterialå¯¼åºæ¨¡æ¿.xlsx"); | 
|---|
|  |  |  | if (!System.IO.File.Exists(templatePath)) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | templatePath = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location) ?? string.Empty, $"Resources/Templates/WmsMaterialå¯¼åºæ¨¡æ¿.xlsx"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var memoryStream = new MemoryStream(); | 
|---|
|  |  |  | await memoryStream.SaveAsByTemplateAsync(templatePath, exportData.Sheets); | 
|---|
|  |  |  | memoryStream.Seek(0L, SeekOrigin.Begin); | 
|---|
|  |  |  | return new FileStreamResult(memoryStream, "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet") { FileDownloadName = $"{exportData.FileName}_{DateTime.Now:yyyyMMddhhmmss}.xlsx" }; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | [HttpGet] | 
|---|
|  |  |  | [Route("Export")] | 
|---|
|  |  |  | public virtual async Task<IActionResult> ExportAsync([FromQuery] GetWmsMaterialInput input) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | input.MaxResultCount = int.MaxValue; | 
|---|
|  |  |  | var exportData = await _wmsMaterialAppService.ExportAsync(input); | 
|---|
|  |  |  | var templatePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, $"Resources/Templates/WmsMaterialå¯¼åºæ¨¡æ¿.xlsx"); | 
|---|
|  |  |  | if (!System.IO.File.Exists(templatePath)) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | templatePath = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location) ?? string.Empty, $"Resources/Templates/WmsMaterialå¯¼åºæ¨¡æ¿.xlsx"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var memoryStream = new MemoryStream(); | 
|---|
|  |  |  | await memoryStream.SaveAsByTemplateAsync(templatePath, exportData.Sheets); | 
|---|
|  |  |  | memoryStream.Seek(0L, SeekOrigin.Begin); | 
|---|
|  |  |  | return new FileStreamResult(memoryStream, "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet") { FileDownloadName = $"{exportData.FileName}_{DateTime.Now:yyyyMMddhhmmss}.xlsx" }; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|