add Excel export functionality for roll call case history
This commit is contained in:
@@ -1,6 +1,12 @@
|
|||||||
using _0_Framework.Excel;
|
using _0_Framework.Excel;
|
||||||
|
using _0_Framework.Application;
|
||||||
|
using CompanyManagment.App.Contracts.RollCall;
|
||||||
using OfficeOpenXml;
|
using OfficeOpenXml;
|
||||||
using OfficeOpenXml.Drawing;
|
using OfficeOpenXml.Drawing;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
|
|
||||||
namespace CompanyManagement.Infrastructure.Excel.RollCall;
|
namespace CompanyManagement.Infrastructure.Excel.RollCall;
|
||||||
|
|
||||||
@@ -308,6 +314,111 @@ public class RollCallExcelGenerator : ExcelGenerator
|
|||||||
return package.GetAsByteArray();
|
return package.GetAsByteArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static byte[] CaseHistoryExcelForEmployee(List<RollCallCaseHistoryDetail> data, string titleId)
|
||||||
|
{
|
||||||
|
if (!Regex.IsMatch(titleId, @"^\d{4}_\d{2}$"))
|
||||||
|
throw new ArgumentException("Invalid titleId format.", nameof(titleId));
|
||||||
|
|
||||||
|
var splitDate = titleId.Split("_");
|
||||||
|
var year = Convert.ToInt32(splitDate.First());
|
||||||
|
var month = Convert.ToInt32(splitDate.Last());
|
||||||
|
|
||||||
|
var startDateFa = $"{year:D4}/{month:D2}/01";
|
||||||
|
var startDate = startDateFa.ToGeorgianDateTime();
|
||||||
|
var endDateFa = startDateFa.FindeEndOfMonth();
|
||||||
|
var endDate = endDateFa.ToGeorgianDateTime();
|
||||||
|
|
||||||
|
var dateRange = (int)(endDate.Date - startDate.Date).TotalDays + 1;
|
||||||
|
var dates = Enumerable.Range(0, dateRange).Select(x => startDate.AddDays(x)).ToList();
|
||||||
|
|
||||||
|
var safeData = data ?? new List<RollCallCaseHistoryDetail>();
|
||||||
|
var first = safeData.FirstOrDefault();
|
||||||
|
var totalWorkingTime = new TimeSpan(safeData.Sum(x => x.TotalWorkingTime.Ticks));
|
||||||
|
|
||||||
|
var viewModel = new CaseHistoryRollCallExcelForEmployeeViewModel
|
||||||
|
{
|
||||||
|
EmployeeId = first?.EmployeeId ?? 0,
|
||||||
|
DateGr = startDate,
|
||||||
|
PersonnelCode = first?.PersonnelCode,
|
||||||
|
EmployeeFullName = first?.EmployeeFullName,
|
||||||
|
PersianMonthName = month.ToFarsiMonthByIntNumber(),
|
||||||
|
PersianYear = year.ToString(),
|
||||||
|
TotalWorkingHoursFa = totalWorkingTime.ToFarsiHoursAndMinutes("-"),
|
||||||
|
TotalWorkingTimeSpan = $"{(int)totalWorkingTime.TotalHours}:{totalWorkingTime.Minutes:00}",
|
||||||
|
RollCalls = dates.Select((date, index) =>
|
||||||
|
{
|
||||||
|
var item = index < safeData.Count ? safeData[index] : null;
|
||||||
|
var records = item?.Records ?? new List<RollCallCaseHistoryDetailRecord>();
|
||||||
|
|
||||||
|
return new RollCallItemForEmployeeExcelViewModel
|
||||||
|
{
|
||||||
|
DateGr = date,
|
||||||
|
DateFa = date.ToFarsi(),
|
||||||
|
DayOfWeekFa = date.DayOfWeek.DayOfWeeKToPersian(),
|
||||||
|
PersonnelCode = item?.PersonnelCode,
|
||||||
|
EmployeeFullName = item?.EmployeeFullName,
|
||||||
|
IsAbsent = item?.Status == RollCallRecordStatus.Absent,
|
||||||
|
HasLeave = item?.Status == RollCallRecordStatus.Leaved,
|
||||||
|
IsHoliday = false,
|
||||||
|
TotalWorkingHours = (item?.TotalWorkingTime ?? TimeSpan.Zero).ToFarsiHoursAndMinutes("-"),
|
||||||
|
StartsItems = JoinRecords(records, r => r.StartTime),
|
||||||
|
EndsItems = JoinRecords(records, r => r.EndTime),
|
||||||
|
EnterTimeDifferences = JoinRecords(records, r => FormatSignedTimeSpan(r.EntryTimeDifference)),
|
||||||
|
ExitTimeDifferences = JoinRecords(records, r => FormatSignedTimeSpan(r.ExitTimeDifference))
|
||||||
|
};
|
||||||
|
}).ToList()
|
||||||
|
};
|
||||||
|
|
||||||
|
return CaseHistoryExcelForEmployee(viewModel);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static byte[] CaseHistoryExcelForOneDay(List<RollCallCaseHistoryDetail> data, string titleId)
|
||||||
|
{
|
||||||
|
if (!Regex.IsMatch(titleId, @"^\d{4}/\d{2}/\d{2}$"))
|
||||||
|
throw new ArgumentException("Invalid titleId format.", nameof(titleId));
|
||||||
|
|
||||||
|
var dateGr = titleId.ToGeorgianDateTime();
|
||||||
|
var safeData = data ?? new List<RollCallCaseHistoryDetail>();
|
||||||
|
|
||||||
|
var viewModel = new CaseHistoryRollCallForOneDayViewModel
|
||||||
|
{
|
||||||
|
DateFa = titleId,
|
||||||
|
DateGr = dateGr,
|
||||||
|
DayOfWeekFa = dateGr.DayOfWeek.DayOfWeeKToPersian(),
|
||||||
|
RollCalls = safeData.Select(item =>
|
||||||
|
{
|
||||||
|
var records = item.Records ?? new List<RollCallCaseHistoryDetailRecord>();
|
||||||
|
return new RollCallItemForOneDayExcelViewModel
|
||||||
|
{
|
||||||
|
EmployeeFullName = item.EmployeeFullName,
|
||||||
|
PersonnelCode = item.PersonnelCode,
|
||||||
|
StartsItems = JoinRecords(records, r => r.StartTime),
|
||||||
|
EndsItems = JoinRecords(records, r => r.EndTime),
|
||||||
|
TotalWorkingHours = item.TotalWorkingTime.ToFarsiHoursAndMinutes("-")
|
||||||
|
};
|
||||||
|
}).ToList()
|
||||||
|
};
|
||||||
|
|
||||||
|
return CaseHistoryExcelForOneDay(viewModel);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string JoinRecords(IEnumerable<RollCallCaseHistoryDetailRecord> records, Func<RollCallCaseHistoryDetailRecord, string> selector)
|
||||||
|
{
|
||||||
|
var safeRecords = records ?? Enumerable.Empty<RollCallCaseHistoryDetailRecord>();
|
||||||
|
var values = safeRecords.Select(selector).Where(x => !string.IsNullOrWhiteSpace(x)).ToList();
|
||||||
|
return values.Count == 0 ? string.Empty : string.Join(Environment.NewLine, values);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string FormatSignedTimeSpan(TimeSpan value)
|
||||||
|
{
|
||||||
|
if (value == TimeSpan.Zero)
|
||||||
|
return "-";
|
||||||
|
|
||||||
|
var abs = value.Duration();
|
||||||
|
var sign = value.Ticks < 0 ? "-" : "+";
|
||||||
|
return $"{(int)abs.TotalHours}:{abs.Minutes:00}{sign}";
|
||||||
|
}
|
||||||
|
|
||||||
private string CalculateExitMinuteDifference(TimeSpan early, TimeSpan late)
|
private string CalculateExitMinuteDifference(TimeSpan early, TimeSpan late)
|
||||||
{
|
{
|
||||||
if (early == TimeSpan.Zero && late == TimeSpan.Zero)
|
if (early == TimeSpan.Zero && late == TimeSpan.Zero)
|
||||||
|
|||||||
@@ -13,6 +13,7 @@
|
|||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\0_Framework\0_Framework.csproj" />
|
<ProjectReference Include="..\0_Framework\0_Framework.csproj" />
|
||||||
<ProjectReference Include="..\Company.Domain\Company.Domain.csproj" />
|
<ProjectReference Include="..\Company.Domain\Company.Domain.csproj" />
|
||||||
|
<ProjectReference Include="..\CompanyManagement.Infrastructure.Excel\CompanyManagement.Infrastructure.Excel.csproj" />
|
||||||
<ProjectReference Include="..\CompanyManagment.App.Contracts\CompanyManagment.App.Contracts.csproj" />
|
<ProjectReference Include="..\CompanyManagment.App.Contracts\CompanyManagment.App.Contracts.csproj" />
|
||||||
<ProjectReference Include="..\CompanyManagment.EFCore\CompanyManagment.EFCore.csproj" />
|
<ProjectReference Include="..\CompanyManagment.EFCore\CompanyManagment.EFCore.csproj" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|||||||
@@ -3,9 +3,11 @@ using System.Collections.Generic;
|
|||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using _0_Framework.Application;
|
using _0_Framework.Application;
|
||||||
using _0_Framework.Domain.CustomizeCheckoutShared.Enums;
|
using _0_Framework.Domain.CustomizeCheckoutShared.Enums;
|
||||||
|
using _0_Framework.Exceptions;
|
||||||
using Company.Domain.CheckoutAgg;
|
using Company.Domain.CheckoutAgg;
|
||||||
using Company.Domain.CustomizeCheckoutAgg;
|
using Company.Domain.CustomizeCheckoutAgg;
|
||||||
using Company.Domain.CustomizeCheckoutTempAgg;
|
using Company.Domain.CustomizeCheckoutTempAgg;
|
||||||
@@ -17,6 +19,7 @@ using Company.Domain.RollCallAgg;
|
|||||||
using Company.Domain.RollCallAgg.DomainService;
|
using Company.Domain.RollCallAgg.DomainService;
|
||||||
using Company.Domain.RollCallEmployeeAgg;
|
using Company.Domain.RollCallEmployeeAgg;
|
||||||
using Company.Domain.WorkshopAgg;
|
using Company.Domain.WorkshopAgg;
|
||||||
|
using CompanyManagement.Infrastructure.Excel.RollCall;
|
||||||
using CompanyManagment.App.Contracts.Checkout;
|
using CompanyManagment.App.Contracts.Checkout;
|
||||||
using CompanyManagment.App.Contracts.Employee;
|
using CompanyManagment.App.Contracts.Employee;
|
||||||
using CompanyManagment.App.Contracts.RollCall;
|
using CompanyManagment.App.Contracts.RollCall;
|
||||||
@@ -874,11 +877,45 @@ public class RollCallApplication : IRollCallApplication
|
|||||||
}
|
}
|
||||||
|
|
||||||
public async Task<RollCallCaseHistoryExcelDto> DownloadCaseHistoryExcel(long workshopId, string titleId,
|
public async Task<RollCallCaseHistoryExcelDto> DownloadCaseHistoryExcel(long workshopId, string titleId,
|
||||||
RollCallCaseHistorySearchModel searchModel)
|
RollCallCaseHistorySearchModel searchModel)
|
||||||
{
|
{
|
||||||
var data = await _rollCallRepository
|
var data = await _rollCallRepository
|
||||||
.GetCaseHistoryDetails(workshopId, titleId, searchModel);
|
.GetCaseHistoryDetails(workshopId, titleId, searchModel);
|
||||||
var workshopFullName = _workshopRepository.(workshopId);
|
string nameSecondPart = "";
|
||||||
byte[] excelBytes = RollCallExcelGenerator.CaseHistoryExcelForOneDay(data);
|
byte[] excelBytes;
|
||||||
|
|
||||||
|
if (Regex.IsMatch(titleId, @"^\d{4}_\d{2}$"))
|
||||||
|
{
|
||||||
|
var splitDate = titleId.Split("_");
|
||||||
|
var year = Convert.ToInt32(splitDate.First());
|
||||||
|
var month = Convert.ToInt32(splitDate.Last());
|
||||||
|
|
||||||
|
var monthName = Convert.ToInt32(month).ToFarsiMonthByIntNumber();
|
||||||
|
nameSecondPart = $"{year}/{monthName}";
|
||||||
|
excelBytes = RollCallExcelGenerator.CaseHistoryExcelForEmployee(data, titleId);
|
||||||
|
}
|
||||||
|
else if (Regex.IsMatch(titleId, @"^\d{4}/\d{2}/\d{2}$"))
|
||||||
|
{
|
||||||
|
var oneDayDate = titleId.ToGeorgianDateTime();
|
||||||
|
nameSecondPart = $" {oneDayDate.DayOfWeek.DayOfWeeKToPersian()}،{titleId}";
|
||||||
|
excelBytes = RollCallExcelGenerator.CaseHistoryExcelForOneDay(data, titleId);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
throw new BadRequestException("شناسه سر تیتر وارد شده نامعتبر است");
|
||||||
|
}
|
||||||
|
|
||||||
|
var workshopFullName = _workshopRepository.Get(workshopId)?.WorkshopFullName ?? "بدون کارگاه";
|
||||||
|
|
||||||
|
var fileName = $"{workshopFullName} - {nameSecondPart}.xlsx";
|
||||||
|
|
||||||
|
var res = new RollCallCaseHistoryExcelDto()
|
||||||
|
{
|
||||||
|
Bytes = excelBytes,
|
||||||
|
MimeType = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
|
||||||
|
FileName = fileName
|
||||||
|
};
|
||||||
|
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -19,7 +19,7 @@
|
|||||||
"sqlDebugging": true,
|
"sqlDebugging": true,
|
||||||
"dotnetRunMessages": "true",
|
"dotnetRunMessages": "true",
|
||||||
"nativeDebugging": true,
|
"nativeDebugging": true,
|
||||||
"applicationUrl": "https://localhost:5004;http://localhost:5003;",
|
"applicationUrl": "https://localhost:5004;http://localhost:5003;https://192.168.0.117:5005",
|
||||||
"jsWebView2Debugging": false,
|
"jsWebView2Debugging": false,
|
||||||
"hotReloadEnabled": true
|
"hotReloadEnabled": true
|
||||||
},
|
},
|
||||||
|
|||||||
Reference in New Issue
Block a user