using Aspose.Cells; using OASystem.API.OAMethodLib; using OASystem.API.OAMethodLib.QiYeWeChatAPI; using OASystem.Domain.Dtos.PersonnelModule; using OASystem.Domain.Entities.PersonnelModule; using OASystem.Domain.ViewModels.PersonnelModule; using OASystem.Domain.ViewModels.QiYeWeChat; using OASystem.Infrastructure.Repositories.PersonnelModule; using System.Diagnostics; using System.Globalization; namespace OASystem.API.Controllers { /// /// 人事模块 /// [Route("api/[controller]/[action]")] public class PersonnelModuleController : ControllerBase { private Result _result; private readonly IMapper _mapper; private readonly decimal _chengDuMinimumWage = 2100.00M; private readonly IQiYeWeChatApiService _qiYeWeChatApiService; private readonly WageSheetRepository _wageSheetRep; private readonly UsersRepository _usersRep; /// /// 初始化 /// /// /// /// /// public PersonnelModuleController(IQiYeWeChatApiService qiYeWeChatApiService,WageSheetRepository wageSheetRep, UsersRepository usersRep, IMapper mapper) { _mapper = mapper; _usersRep = usersRep; _qiYeWeChatApiService = qiYeWeChatApiService; _wageSheetRep = wageSheetRep; _result = new Result(); } #region 工资表单 /// /// 工资 月列表 /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task GetWageSheetMonth() { string sql = string.Format("Select * From Pm_WageIssueWorkingDay Where IsDel = 0 Order By YearMonth Desc"); var data = await _wageSheetRep._sqlSugar.SqlQueryable(sql).ToListAsync(); return Ok(JsonView(true, "查询成功!", data)); } /// /// 工资 工作日信息 /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task GetWageSheetMonthWorkdays(string startDt, string endDt ) { //参数处理 string dtFormat = "yyyy-MM-dd"; DateTime startDt1, endDt1; bool startDtIsValid = DateTime.TryParseExact(startDt, dtFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out startDt1); bool endDtIsValid = DateTime.TryParseExact(endDt, dtFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out endDt1); if (!startDtIsValid) return Ok(JsonView(false, "开始日期格式错误!正确时间格式:yyyy-MM-dd ")); if (!endDtIsValid) return Ok(JsonView(false, "结束格式错误!正确时间格式:yyyy-MM-dd ")); string sql = string.Format(@"Select * From Sys_Calendar Where Isdel = 0 And Dt between '{0}' And '{1}'", startDt, endDt); var data = await _wageSheetRep._sqlSugar.SqlQueryable(sql).ToListAsync(); return Ok(JsonView(true, "查询成功!", data)); } /// /// 工资 工作日信息 /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task GetWageSheetMonthWorkdaysAddOrEdit(WageSheetMonthWorkdaysAddOrEditDto dto) { //参数处理 string yearFormat = "yyyy-MM"; string dtFormat = "yyyy-MM-dd"; DateTime yearDt, startDt1, endDt1; bool yearDtIsValid = DateTime.TryParseExact(dto.YearMonth, yearFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out yearDt); bool startDtIsValid = DateTime.TryParseExact(dto.StartDate, dtFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out startDt1); bool endDtIsValid = DateTime.TryParseExact(dto.EndDate, dtFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out endDt1); if (!yearDtIsValid) return Ok(JsonView(false, "年月日期格式错误!正确时间格式:yyyy-MM ")); if (!startDtIsValid) return Ok(JsonView(false, "开始日期格式错误!正确时间格式:yyyy-MM-dd ")); if (!endDtIsValid) return Ok(JsonView(false, "结束格式错误!正确时间格式:yyyy-MM-dd ")); int res = -1; #region 处理数据 Pm_WageIssueWorkingDay pm_WageIssueWorkingDay = new Pm_WageIssueWorkingDay(); pm_WageIssueWorkingDay = _mapper.Map(dto); List sys_Calendars = new List(); sys_Calendars = _mapper.Map>(dto.CalendarInfos); pm_WageIssueWorkingDay.Workdays = sys_Calendars.Where(it => it.IsWorkDays == true).ToList().Count(); foreach (var item in sys_Calendars) { item.Remark = pm_WageIssueWorkingDay.Remark; item.CreateUserId = pm_WageIssueWorkingDay.CreateUserId; item.CreateTime = pm_WageIssueWorkingDay.CreateTime; } #endregion var _sqlSugar = _wageSheetRep._sqlSugar; if (dto.Status == 1) { string sql = string.Format("Select * From Pm_WageIssueWorkingDay Where IsDel = 0 and YearMonth='{0}'",dto.YearMonth); var data = await _sqlSugar.SqlQueryable(sql).FirstAsync(); if (data != null) { return Ok(JsonView(false, dto.YearMonth + "日期数据已存在,请前往修改!")); } _sqlSugar.BeginTran(); int add1 = await _sqlSugar.Insertable< Pm_WageIssueWorkingDay>(pm_WageIssueWorkingDay).ExecuteCommandAsync(); if (add1 > 0) res = 0; int add2 = await _sqlSugar.Insertable(sys_Calendars).ExecuteCommandAsync(); if (add2 > 0) res = 0; _sqlSugar.CommitTran(); } else if (dto.Status == 1) { _sqlSugar.BeginTran(); int upd1 = await _sqlSugar.Updateable< Pm_WageIssueWorkingDay >(pm_WageIssueWorkingDay) .UpdateColumns(it => new { it.YearMonth,it.StartDate,it.EndDate,it.Workdays,it.DeleteTime }) .ExecuteCommandAsync(); if (upd1 > 0) res = 0; int upd2 = await _sqlSugar.Updateable(sys_Calendars) .UpdateColumns(it => new { it.Dt, it.IsWorkDays, it.IsHoliDay, it.HoliDayName }) .ExecuteCommandAsync(); if (upd2 > 0) res = 0; _sqlSugar.CommitTran(); } else return Ok(JsonView(false, "Status 请传入正确的状态码 1 添加 2 修改")); if (res == 0) return Ok(JsonView(true, "操作成功!")); return Ok(JsonView(false, "操作失败!")); } /// /// 工资表单 基础数据源 /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task GetWageSheetBasicsDataSource() { string companySql = string.Format("Select * From Sys_Company Where IsDel = 0"); var compnayData = await _wageSheetRep._sqlSugar.SqlQueryable(companySql).ToListAsync(); string depSql = string.Format("Select * From Sys_Department Where IsDel = 0"); var depData = await _wageSheetRep._sqlSugar.SqlQueryable(depSql).ToListAsync(); //获取OA系统内所有用户 var nameData = await _usersRep.GetUserNameList(1); if (nameData.Code != 0) { return Ok(JsonView(false, nameData.Msg)); } var data = new { compnayData = compnayData, depData = depData, userNames = nameData.Data }; return Ok(JsonView(true, "查询成功!", data)); } /// /// 获取工资发放月份 /// /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task GetWageYaerMonths(WageYearDto dto) { string sql = string.Format(@"Select * From Pm_WageIssueWorkingDay Where Isdel = 0 And YearMonth Like '%{0}%' Order By YearMonth Asc", dto.Year); var data = await _wageSheetRep._sqlSugar.SqlQueryable(sql).ToListAsync(); return Ok(JsonView(true,"操作成功!", data)); } /// /// 获取工资表单 /// /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task GetWageSheetList(WageSheetListDto dto) { //参数处理 string ymFormat = "yyyy-MM"; DateTime yearMonthDt; bool yearMonthDttIsValid = DateTime.TryParseExact(dto.YearMonth, ymFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out yearMonthDt); if (!yearMonthDttIsValid) { _result.Msg = "年月格式错误!正确时间格式:yyyy-MM "; return Ok(JsonView(false, _result.Msg)); } //获取月工资数据 string yearMonth = yearMonthDt.ToString("yyyy-MM"); if (dto.PortType == 1) { _result = await _wageSheetRep.Get_WageSheet_ListByYearMonthAsync(yearMonth); if (_result.Code != 0) { return Ok(JsonView(false, _result.Msg)); } } else if (dto.PortType == 2) { } else if (dto.PortType == 3) { } else { return Ok(JsonView(false, "请选择正确的端口参数")); } return Ok(JsonView(true, _result.Msg, _result.Data)); } /// /// 获取工资 详情 /// /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task GetWageSheetById(WageSheetInfoDto dto) { if (dto.PortType == 1) { _result = await _wageSheetRep.Get_WageSheet_InfoByIdAsync(dto.Id); if (_result.Code != 0) { return Ok(JsonView(false, _result.Msg)); } } else if (dto.PortType == 2) { } else if (dto.PortType == 3) { } else { return Ok(JsonView(false, "请选择正确的端口参数")); } return Ok(JsonView(true, _result.Msg, _result.Data)); } /// /// 人事模块 工资表单 删除 /// /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task PostWageSheetDel(WageDelDto dto) { try { _result = await _wageSheetRep.Post_WageSheet_DelAsync(dto); if (_result.Code != 0) { return Ok(JsonView(false, _result.Msg)); } } catch (Exception ex) { return Ok(JsonView(false, ex.Message)); } return Ok(JsonView(true, _result.Msg, _result.Data)); } /// /// 人事模块 工资表单 添加 Or 修改 /// /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task PostWageSheetAddOrEdit(WageAddOrEditDto dto) { try { Pm_WageSheet pm_WageSheet = new Pm_WageSheet(); pm_WageSheet = _mapper.Map(dto); pm_WageSheet.LastUpdateUserId = dto.CreateUserId; #region 计算工资 //月工资 decimal salary = pm_WageSheet.Basic + pm_WageSheet.Floats + pm_WageSheet.PostAllowance + pm_WageSheet.InformationSecurityFee + pm_WageSheet.OtherSubsidies; //扣款合计 decimal totalDeduction = pm_WageSheet.SickLeave + pm_WageSheet.SomethingFalse + pm_WageSheet.LateTo + pm_WageSheet.LeaveEarly + pm_WageSheet.Absenteeism + pm_WageSheet.NotPunch + pm_WageSheet.ReservedFunds + pm_WageSheet.WithholdingInsurance + pm_WageSheet.OtherDeductions + pm_WageSheet.OtherDeductions; //实发合计 不含个税 if (pm_WageSheet.RegularDays >= pm_WageSheet.WorkDays ) { pm_WageSheet.RegularDays = pm_WageSheet.WorkDays; salary = salary + pm_WageSheet.Mealsupplement; } else { if (dto.UserId == 21) //21==张海麟 { salary = salary + pm_WageSheet.Mealsupplement; } else { salary = salary = (PayrollComputation.ConvertToDecimal(salary / pm_WageSheet.WorkDays) * pm_WageSheet.RegularDays) + pm_WageSheet.Mealsupplement; } } decimal actualTotal = salary - totalDeduction; pm_WageSheet.Should = salary; pm_WageSheet.TotalDeductions = totalDeduction; pm_WageSheet.TotalRealHair = actualTotal; pm_WageSheet.AfterTax = actualTotal - pm_WageSheet.WithholdingTax; #endregion _result = await _wageSheetRep.Post_WageSheet_AddOrEditAsync(dto, pm_WageSheet); if (_result.Code != 0) { return Ok(JsonView(false, _result.Msg)); } } catch (Exception ex) { return Ok(JsonView(false, ex.Message)); } return Ok(JsonView(true, _result.Msg, _result.Data)); } /// /// 计算工资 /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task SalaryCalculatorAsync(SalaryCalculatorDto dto) { Result result = new Result(); Stopwatch sw = new Stopwatch(); sw.Start(); //参数处理 string ymFormat = "yyyy-MM"; string dtFormat = "yyyy-MM-dd"; DateTime yearMonthDt,startDt,endDt; bool yearMonthDtIsValid = DateTime.TryParseExact(dto.yearMonth, ymFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out yearMonthDt); bool startDtIsValid = DateTime.TryParseExact(dto.startDt, dtFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out startDt); bool endDtIsValid = DateTime.TryParseExact(dto.endDt, dtFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out endDt); if (!yearMonthDtIsValid) return Ok(JsonView(false, "年月格式错误!正确时间格式:yyyy-MM ")); if (!startDtIsValid) return Ok(JsonView(false, "开始日期格式错误!正确时间格式:yyyy-MM-dd ")); if (!endDtIsValid) return Ok(JsonView(false, "结束格式错误!正确时间格式:yyyy-MM-dd ")); string thisYearMonth = dto.yearMonth; string preYearMonth = yearMonthDt.AddMonths(-1).ToString("yyyy-MM"); //计算本月工资起止时间 比如是2月的1号-28号,那就是2月1号的零点到3月1号的零点 DateTime thisStartDt = startDt; DateTime thisEndDt = endDt; // //本月工资是否有数据 有数据则不计算 result = await _wageSheetRep.Get_WageSheet_ListByYearMonthAsync(thisYearMonth); if (result.Code == 0 ) { return Ok(JsonView(false, thisYearMonth + " 工资数据已存在,若无人员工资请手动添加!")); } //获取上个月工资信息 List preWageSheetItems = await _wageSheetRep._sqlSugar.Queryable().Where(it => it.IsDel == 0 && it.YearMonth == preYearMonth).ToListAsync(); preWageSheetItems = preWageSheetItems.OrderBy(it => it.UserId).ToList(); if (preWageSheetItems.Count <= 0) { return Ok(JsonView(false, thisYearMonth + " 上月工资数据不存在,请手动添加!")); } //处理上个月同月同人 多条数据 List preWageSheetItems1 = new List(); preWageSheetItems1 = preWageSheetItems.GroupBy(it => new { it.YearMonth,it.UserId }) .Select(it => it.FirstOrDefault(item => item.Basic !=0)) .ToList(); //获取OA系统内所有用户 var nameData = await _usersRep.GetUserNameList(1); List? userNames = nameData.Data; List wageSheets = new List(); _result = await PayrollComputation.SalaryCalculatorAsync(preWageSheetItems1, userNames,dto.UserId, thisYearMonth, thisStartDt,thisEndDt); #region 批量添加 if (_result.Code != 0) { return Ok(JsonView(false, _result.Msg)); } wageSheets = _result.Data; var add = await _wageSheetRep._sqlSugar.Insertable(wageSheets).ExecuteCommandAsync(); if (add <= 0) { return Ok(JsonView(false, "操作失败!")); } #endregion #region 处理返回数据 //List wageSheetItems = new List(); //wageSheetItems = _mapper.Map>(wageSheets); //wageSheetItems = wageSheetItems.Select(it => // { // UserNameView? uName1 = new UserNameView(); // UserNameView? uName2 = new UserNameView(); // uName1 = userNames.Where(it1 => it.UserId == it1.Id).FirstOrDefault(); // if (uName1 != null) it.Name = uName1.CnName; // uName2 = userNames.Where(it1 => it.LastUpdateUserId == it1.Id).FirstOrDefault(); // if (uName2 != null) it.LastUpdateUserName = uName2.CnName; // return it; } // ).ToList(); #endregion sw.Stop(); return Ok(JsonView(true, "操作成功! 耗时:" + (sw.ElapsedMilliseconds/1000) + "s")); } /// /// 计算工资 By YearMonth And UserId /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task SalaryCalculatorSingleAsync(SalaryCalculatorSingleDto dto) { Result result = new Result(); Stopwatch sw = new Stopwatch(); sw.Start(); //参数处理 string ymFormat = "yyyy-MM"; string ymdFormat = "yyyy-MM-dd"; DateTime yearMonthDt,startDt,endDt; bool yearMonthDtIsValid = DateTime.TryParseExact(dto.YearMonth, ymFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out yearMonthDt); bool startDtIsValid = DateTime.TryParseExact(dto.StartDate, ymdFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out startDt); bool endDtIsValid = DateTime.TryParseExact(dto.EndDate, ymdFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out endDt); if (!yearMonthDtIsValid) return Ok(JsonView(false, "年月格式错误!正确时间格式:yyyy-MM ")); if (!startDtIsValid) return Ok(JsonView(false, "开始时间格式错误!正确时间格式:yyyy-MM-dd ")); if (!yearMonthDtIsValid) return Ok(JsonView(false, "结束时间格式错误!正确时间格式:yyyy-MM-dd ")); List wageSheets = new List(); Pm_WageSheet wageSheet = _mapper.Map(dto); wageSheets.Add(wageSheet); //获取OA系统内所有用户 var nameData = await _usersRep.GetUserNameList(1); List? userNames = nameData.Data; _result = await PayrollComputation.SalaryCalculatorAsync(wageSheets, userNames, dto.UserId, dto.YearMonth, startDt, endDt); if (_result.Code != 0) { return Ok(JsonView(false, _result.Msg)); } List wageSheets1 = new List(); wageSheets1 = _result.Data; #region 处理返回数据 List wageSheetItems = new List(); wageSheetItems = _mapper.Map>(wageSheets1); wageSheetItems = wageSheetItems.Select(it => { UserNameView? uName1 = new UserNameView(); UserNameView? uName2 = new UserNameView(); uName1 = userNames.Where(it1 => it.UserId == it1.Id).FirstOrDefault(); if (uName1 != null) it.Name = uName1.CnName; uName2 = userNames.Where(it1 => it.LastUpdateUserId == it1.Id).FirstOrDefault(); if (uName2 != null) it.LastUpdateUserName = uName2.CnName; return it; } ).ToList(); #endregion sw.Stop(); return Ok(JsonView(true, "操作成功!耗时:" + (sw.ElapsedMilliseconds / 1000) + "s", wageSheetItems[0])); } /// /// 导出工资单 /// /// [HttpPost] [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)] public async Task ExportWageCard(string yearMonth) { Result result = new Result(); Stopwatch sw = new Stopwatch(); sw.Start(); //参数处理 string ymFormat = "yyyy-MM"; DateTime yearMonthDt; bool yearMonthDtIsValid = DateTime.TryParseExact(yearMonth, ymFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out yearMonthDt); if (!yearMonthDtIsValid) return Ok(JsonView(false, "年月格式错误!正确时间格式:yyyy-MM ")); //公司部门 string sql = string.Format(@"Select row_number() over(order by pm_ws.Id) as Row_Number, sc.Id as CompanyId,sc.CompanyName,sd.Id as DepId,sd.DepName, sys_u1.CnName Name,sys_u2.CnName LastUpdateUserName,pm_ws.* From Pm_WageSheet pm_ws Left Join Sys_Users sys_u1 On pm_ws.UserId = sys_u1.Id Left Join Sys_Users sys_u2 On pm_ws.LastUpdateUserId = sys_u2.Id Left Join Sys_Company sc On sys_u1.companyId = sc.Id Left Join Sys_Department sd On sys_u1.DepId = sd.Id Where pm_ws.IsDel = 0 And pm_ws.YearMonth = '{0}'", yearMonth); var wageSheetList = await _wageSheetRep._sqlSugar.SqlQueryable(sql).ToListAsync(); if (wageSheetList.Count <= 0) { return Ok(JsonView(false, yearMonth + "暂无工资数据!")); } decimal SumPrice = 0.00M; foreach (var item in wageSheetList) { SumPrice += item.AfterTax; } WorkbookDesigner designer = new WorkbookDesigner(); designer.Workbook = new Workbook(AppSettingsHelper.Get("ExcelBasePath") + "Template/工资详细清单.xlsx"); designer.Workbook.Worksheets[0].Name = yearMonth +" 工资单"; designer.SetDataSource("WageSheet", wageSheetList); designer.SetDataSource("YearMonth", yearMonth); designer.SetDataSource("StartEndDt", wageSheetList[0].StartDate +" - "+ wageSheetList[0].EndDate); designer.SetDataSource("WorkDays", wageSheetList[0].WorkDays); designer.SetDataSource("SumPrice", SumPrice); designer.SetDataSource("WageSheetTitle", "工资单");// designer.Process(); string fileName = "WageCard/" + yearMonth + "_工资单_" + DateTime.Now.ToString("yyyyMMddHHmmss") + ".xlsx"; string path = AppSettingsHelper.Get("ExcelBasePath"); designer.Workbook.Save(path + fileName); designer = null; string excelPath = AppSettingsHelper.Get("ExcelFtpPath") + fileName; string url = AppSettingsHelper.Get("ExcelBaseUrl"); string fileUrl = url + excelPath; sw.Stop(); return Ok(JsonView(true, "操作成功!耗时:" + (sw.ElapsedMilliseconds / 1000) + "s", new { FileUrl = fileUrl })); } #endregion } }