using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; using System.Threading; using System.Threading.Tasks; using AsbCloudApp.Requests.ExportOptions; using AsbCloudApp.Services.Export; using AsbCloudInfrastructure.Services.ExcelServices.Templates; using ClosedXML.Excel; using Mapster; namespace AsbCloudInfrastructure.Services.ExcelServices; public abstract class ExcelExportService : IExportService where TOptions : IExportOptionsRequest where TTemplate : class, ITemplateParameters, new() { protected TTemplate TemplateParameters => new(); protected abstract Task BuildFileNameAsync(TOptions options, CancellationToken token); protected abstract Task> GetDtosAsync(TOptions options, CancellationToken token); public async Task<(string FileName, Stream File)> ExportAsync(TOptions options, CancellationToken token) { var dtos = await GetDtosAsync(options, token); var fileName = await BuildFileNameAsync(options, token); var file = BuildFile(dtos); return (fileName, file); } private Stream BuildFile(IEnumerable dtos) { using var template = GetTemplateFile(); using var workbook = new XLWorkbook(template); AddDtosToWorkbook(workbook, dtos); var memoryStream = new MemoryStream(); workbook.SaveAs(memoryStream, new SaveOptions { }); memoryStream.Seek(0, SeekOrigin.Begin); return memoryStream; } private void AddDtosToWorkbook(XLWorkbook workbook, IEnumerable dtos) { var dtosToArray = dtos.ToArray(); if (!dtosToArray.Any()) return; var sheet = workbook.GetWorksheet(TemplateParameters.SheetName); for (var i = 0; i < dtosToArray.Length; i++) { var row = sheet.Row(1 + i + TemplateParameters.HeaderRowsCount); AddRow(row, dtosToArray[i]); } } private void AddRow(IXLRow xlRow, TDto dto) { var properties = dto.Adapt>(); foreach (var (name, cellValue) in properties) { if (TemplateParameters.Cells.TryGetValue(name, out var cell)) xlRow.Cell(cell.ColumnNumber).SetCellValue(cellValue); } } private Stream GetTemplateFile() => Assembly.GetExecutingAssembly().GetTemplateCopyStream(TemplateParameters.FileName) ?? throw new ArgumentNullException($"Файл '{TemplateParameters.FileName}' не найден"); }