瀏覽代碼

解决冲突

wangh 1 年之前
父節點
當前提交
b9ff9b3b32

+ 0 - 1
OASystem/EntitySync/Program.cs

@@ -110,6 +110,5 @@ db.CodeFirst.SetStringDefaultLength(50).BackupTable().InitTables(new Type[]
     //typeof(Pm_WageSheet)       //人事模块 工资表单
     //typeof(Pm_WageSheet)       //人事模块 工资表单
     //typeof(Grp_VisaProgressCustomer),
     //typeof(Grp_VisaProgressCustomer),
     //typeof(Grp_VisaProgressCustomerPicture)
     //typeof(Grp_VisaProgressCustomerPicture)
-    typeof(Grp_Ommission)
 });
 });
 Console.WriteLine("数据库结构同步完成!");
 Console.WriteLine("数据库结构同步完成!");

+ 114 - 1
OASystem/OASystem.Api/Controllers/GroupsController.cs

@@ -25,6 +25,7 @@ using OASystem.Domain.Dtos.CRM;
 using System.Diagnostics;
 using System.Diagnostics;
 using MathNet.Numerics.Statistics.Mcmc;
 using MathNet.Numerics.Statistics.Mcmc;
 using AlibabaCloud.OpenApiClient.Models;
 using AlibabaCloud.OpenApiClient.Models;
+using System;
 
 
 namespace OASystem.API.Controllers
 namespace OASystem.API.Controllers
 {
 {
@@ -483,6 +484,118 @@ namespace OASystem.API.Controllers
         }
         }
 
 
 
 
+        /// <summary>
+        ///  IOS获取团组签证拍照上传进度03(相册)
+        /// </summary>
+        /// <returns></returns>
+        [HttpPost]
+        [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
+        public async Task<ActionResult> PostIOSVisaProgressImageList(IOS_VisaImageListDto dto)
+        {
+            if (dto == null)
+            {
+                return Ok(JsonView(false, "请求错误:"));
+            }
+
+            List<VisaProgressImageView> list = _delegationVisaRep.GetVisaProgressImageList(dto.visaProgressCustomerId, dto.picType);
+
+            return Ok(JsonView(list));
+        }
+
+
+        /// <summary>
+        /// IOS获取团组签证拍照上传进度04(图片上传)
+        /// </summary>
+        /// <param name="dto"></param>
+        /// <returns></returns>
+        [HttpPost]
+        [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
+        public async Task<ActionResult> PostIOSVisaProgressUploadImage(IOS_VisaUploadImageDto dto)
+        {
+
+            //string result = decodeBase64ToImage(dto.base64DataURL, dto.imageName);
+
+            //if (!string.IsNullOrEmpty(result))
+            //{
+
+            //}
+            //else {
+            //    return Ok(JsonView(false, "上传失败"));
+            //}
+
+            DateTime dt1970 = new DateTime(1970, 1, 1, 0, 0, 0, 0);
+            int sucNum = 0;
+            foreach (var item in dto.base64DataList)
+            {
+                string imageName = dto.imageName + ((DateTime.Now.Ticks - dt1970.Ticks) / 10000).ToString();
+                string result = decodeBase64ToImage(item, imageName);
+
+                if (!string.IsNullOrEmpty(result))
+                {
+                    Grp_VisaProgressCustomerPicture pic = new Grp_VisaProgressCustomerPicture();
+                    pic.CreateUserId = dto.CreateUserId;
+                    pic.PicName = imageName;
+                    pic.PicPath = result;
+                    pic.VisaProgressCustomerId = dto.visaProgressCustomerId;
+
+                    int insertResult = await _delegationVisaRep.AddAsync<Grp_VisaProgressCustomerPicture>(pic);
+                    if (insertResult > 0)
+                    {
+                        sucNum++;
+                    }
+                }
+            }
+
+            string msg = string.Format(@"成功上传{0}张", sucNum);
+
+            return Ok(JsonView(true, msg));
+        }
+
+        private string decodeBase64ToImage(string base64DataURL, string imgName)
+        {
+            string filename = "";//声明一个string类型的相对路径
+
+            String base64 = base64DataURL.Substring(base64DataURL.IndexOf(",") + 1);      //将‘,’以前的多余字符串删除
+            System.Drawing.Bitmap bitmap = null;//定义一个Bitmap对象,接收转换完成的图片
+
+            try//会有异常抛出,try,catch一下
+            {
+
+                byte[] arr = Convert.FromBase64String(base64);//将纯净资源Base64转换成等效的8位无符号整形数组
+
+                System.IO.MemoryStream ms = new System.IO.MemoryStream(arr);//转换成无法调整大小的MemoryStream对象
+                bitmap = new System.Drawing.Bitmap(ms);//将MemoryStream对象转换成Bitmap对象
+
+                var fileDir = AppSettingsHelper.Get("VisaProgressImageFtpPath");
+                //文件名称
+                filename = "VisaProgress_" + DateTime.Now.ToString("yyyyMMddHHmmss") + "_" + imgName + ".jpg";//所要保存的相对路径及名字
+                //上传的文件的路径
+                string filePath = "";
+                if (!Directory.Exists(fileDir))
+                {
+                    Directory.CreateDirectory(fileDir);
+                }
+                //上传的文件的路径
+                filePath = fileDir + filename;
+
+                //string url = HttpRuntime.AppDomainAppPath.ToString();
+                //string tmpRootDir = System.Web.HttpContext.Current.Server.MapPath(System.Web.HttpContext.Current.Request.ApplicationPath.ToString()); //获取程序根目录 
+                //string imagesurl2 = tmpRootDir + filename; //转换成绝对路径 
+                bitmap.Save(filePath, System.Drawing.Imaging.ImageFormat.Jpeg);//保存到服务器路径
+                //bitmap.Save(filePath + ".bmp", System.Drawing.Imaging.ImageFormat.Bmp);
+                //bitmap.Save(filePath + ".gif", System.Drawing.Imaging.ImageFormat.Gif);
+                //bitmap.Save(filePath + ".png", System.Drawing.Imaging.ImageFormat.Png);
+                ms.Close();//关闭当前流,并释放所有与之关联的资源
+                bitmap.Dispose();
+            }
+            catch (Exception e)
+            {
+                string massage = e.Message;
+            }
+            return filename;//返回相对路径
+        }
+
+
         #endregion
         #endregion
 
 
         #region 团组任务分配
         #region 团组任务分配
@@ -906,7 +1019,7 @@ namespace OASystem.API.Controllers
         {
         {
             try
             try
             {
             {
-                
+
 
 
                 Result groupData = await _airTicketResRep.OpAirTicketRes(dto);
                 Result groupData = await _airTicketResRep.OpAirTicketRes(dto);
                 if (groupData.Code != 0)
                 if (groupData.Code != 0)

+ 148 - 17
OASystem/OASystem.Api/Controllers/PersonnelModuleController.cs

@@ -1,28 +1,13 @@
 using Aspose.Cells;
 using Aspose.Cells;
-using Autofac.Diagnostics;
-using MathNet.Numerics.Statistics.Mcmc;
-using Microsoft.AspNetCore.Mvc;
-using Microsoft.AspNetCore.Mvc.RazorPages;
 using OASystem.API.OAMethodLib;
 using OASystem.API.OAMethodLib;
 using OASystem.API.OAMethodLib.QiYeWeChatAPI;
 using OASystem.API.OAMethodLib.QiYeWeChatAPI;
-using OASystem.Domain;
 using OASystem.Domain.Dtos.PersonnelModule;
 using OASystem.Domain.Dtos.PersonnelModule;
-using OASystem.Domain.Dtos.QiYeWeChat;
-using OASystem.Domain.Entities.Business;
-using OASystem.Domain.Entities.Groups;
 using OASystem.Domain.Entities.PersonnelModule;
 using OASystem.Domain.Entities.PersonnelModule;
-using OASystem.Domain.ViewModels.Business;
-using OASystem.Domain.ViewModels.JuHeExchangeRate;
 using OASystem.Domain.ViewModels.PersonnelModule;
 using OASystem.Domain.ViewModels.PersonnelModule;
 using OASystem.Domain.ViewModels.QiYeWeChat;
 using OASystem.Domain.ViewModels.QiYeWeChat;
 using OASystem.Infrastructure.Repositories.PersonnelModule;
 using OASystem.Infrastructure.Repositories.PersonnelModule;
-using SqlSugar;
-using StackExchange.Redis;
-using System.Collections.Generic;
 using System.Diagnostics;
 using System.Diagnostics;
 using System.Globalization;
 using System.Globalization;
-using System.IO;
-using System.Web;
 
 
 namespace OASystem.API.Controllers
 namespace OASystem.API.Controllers
 {
 {
@@ -58,6 +43,133 @@ namespace OASystem.API.Controllers
 
 
         #region 工资表单
         #region 工资表单
 
 
+
+        /// <summary>
+        /// 工资 月列表
+        /// </summary>
+        /// <returns></returns>
+        [HttpPost]
+        [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
+        public async Task<IActionResult> GetWageSheetMonth()
+        {
+            string sql = string.Format("Select * From  Pm_WageIssueWorkingDay Where IsDel = 0 Order By YearMonth Desc");
+            var data = await _wageSheetRep._sqlSugar.SqlQueryable<WageSheetMonthView>(sql).ToListAsync();
+
+            return Ok(JsonView(true, "查询成功!", data));
+        }
+
+        /// <summary>
+        /// 工资 工作日信息
+        /// </summary>
+        /// <returns></returns>
+        [HttpPost]
+        [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
+        public async Task<IActionResult> 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<CalendarInfoView>(sql).ToListAsync();
+
+            return Ok(JsonView(true, "查询成功!", data));
+
+        }
+
+        /// <summary>
+        /// 工资 工作日信息
+        /// </summary>
+        /// <returns></returns>
+        [HttpPost]
+        [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
+        public async Task<IActionResult> 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<Pm_WageIssueWorkingDay>(dto);
+            List<Sys_Calendar> sys_Calendars = new List<Sys_Calendar>();
+            sys_Calendars = _mapper.Map<List<Sys_Calendar>>(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<WageSheetMonthView>(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_Calendar>(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_Calendar>(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, "操作失败!"));
+        }
+
+
         /// <summary>
         /// <summary>
         /// 工资表单 基础数据源
         /// 工资表单 基础数据源
         /// </summary>
         /// </summary>
@@ -90,6 +202,25 @@ namespace OASystem.API.Controllers
             return Ok(JsonView(true, "查询成功!", data));
             return Ok(JsonView(true, "查询成功!", data));
         }
         }
 
 
+        /// <summary>
+        /// 获取工资发放月份
+        /// </summary>
+        /// <param name="dto"></param>
+        /// <returns></returns>
+        [HttpPost]
+        [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
+        public async Task<IActionResult> 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<WageYearMonthView>(sql).ToListAsync();
+            
+           
+            return Ok(JsonView(true,"操作成功!", data));
+        }
+
         /// <summary>
         /// <summary>
         /// 获取工资表单
         /// 获取工资表单
         /// </summary>
         /// </summary>
@@ -288,7 +419,7 @@ namespace OASystem.API.Controllers
 
 
             //计算本月工资起止时间 比如是2月的1号-28号,那就是2月1号的零点到3月1号的零点 
             //计算本月工资起止时间 比如是2月的1号-28号,那就是2月1号的零点到3月1号的零点 
             DateTime thisStartDt = startDt;
             DateTime thisStartDt = startDt;
-            DateTime thisEndDt = endDt.AddDays(1); //
+            DateTime thisEndDt = endDt; //
 
 
             //本月工资是否有数据 有数据则不计算
             //本月工资是否有数据 有数据则不计算
             result = await _wageSheetRep.Get_WageSheet_ListByYearMonthAsync(thisYearMonth);
             result = await _wageSheetRep.Get_WageSheet_ListByYearMonthAsync(thisYearMonth);
@@ -392,7 +523,7 @@ namespace OASystem.API.Controllers
             var nameData = await _usersRep.GetUserNameList(1);
             var nameData = await _usersRep.GetUserNameList(1);
             List<UserNameView>? userNames = nameData.Data;
             List<UserNameView>? userNames = nameData.Data;
 
 
-            _result = await PayrollComputation.SalaryCalculatorAsync(wageSheets, userNames, dto.UserId, dto.YearMonth, startDt, endDt.AddDays(1));
+            _result = await PayrollComputation.SalaryCalculatorAsync(wageSheets, userNames, dto.UserId, dto.YearMonth, startDt, endDt);
 
 
             if (_result.Code != 0)
             if (_result.Code != 0)
             {
             {

+ 538 - 117
OASystem/OASystem.Api/OAMethodLib/PayrollComputation.cs

@@ -34,7 +34,7 @@ namespace OASystem.API.OAMethodLib
         /// <param name="startDt"></param>
         /// <param name="startDt"></param>
         /// <param name="endDt"></param>
         /// <param name="endDt"></param>
         /// <returns></returns>
         /// <returns></returns>
-        public static async Task<Result> SalaryCalculatorAsync(
+        public static async Task<Result> SalaryCalculatorAsync1(
             List<Pm_WageSheet> pm_WageSheetDattaSources, List<UserNameView> userNames, int userId, string thisYearMonth, DateTime startDt, DateTime endDt)
             List<Pm_WageSheet> pm_WageSheetDattaSources, List<UserNameView> userNames, int userId, string thisYearMonth, DateTime startDt, DateTime endDt)
         {
         {
             if (pm_WageSheetDattaSources.Count <= 0)
             if (pm_WageSheetDattaSources.Count <= 0)
@@ -51,7 +51,7 @@ namespace OASystem.API.OAMethodLib
 
 
             //计算时间段内工作日
             //计算时间段内工作日
 
 
-            int work_days = GetWorkDays(startDt, endDt);
+            int work_days = await GetWorkDays(thisYearMonth);
 
 
             //获取所有打卡数据
             //获取所有打卡数据
             CheckInView checkIn = await _qiYeWeChatApiService.GetCheckin_MonthDataAsync(startDt, endDt); //时间段内所有 打卡数据
             CheckInView checkIn = await _qiYeWeChatApiService.GetCheckin_MonthDataAsync(startDt, endDt); //时间段内所有 打卡数据
@@ -296,6 +296,7 @@ namespace OASystem.API.OAMethodLib
                         #endregion
                         #endregion
 
 
                         Ex_Items ex_Items_dk = new Ex_Items() { Type = "打卡" };    //打卡
                         Ex_Items ex_Items_dk = new Ex_Items() { Type = "打卡" };    //打卡
+
                         List<Ex_Item> ex_reissuecard_Items = new List<Ex_Item>();
                         List<Ex_Item> ex_reissuecard_Items = new List<Ex_Item>();
                         #region 打卡补卡 补卡次数 处理
                         #region 打卡补卡 补卡次数 处理
 
 
@@ -737,7 +738,7 @@ namespace OASystem.API.OAMethodLib
         /// <param name="startDt"></param>
         /// <param name="startDt"></param>
         /// <param name="endDt"></param>
         /// <param name="endDt"></param>
         /// <returns></returns>
         /// <returns></returns>
-        public static async Task<Result> SalaryCalculatorAsync1(
+        public static async Task<Result> SalaryCalculatorAsync(
             List<Pm_WageSheet> pm_WageSheetDattaSources, List<UserNameView> userNames, int userId, string thisYearMonth, DateTime startDt, DateTime endDt)
             List<Pm_WageSheet> pm_WageSheetDattaSources, List<UserNameView> userNames, int userId, string thisYearMonth, DateTime startDt, DateTime endDt)
         {
         {
             if (pm_WageSheetDattaSources.Count <= 0)
             if (pm_WageSheetDattaSources.Count <= 0)
@@ -753,7 +754,12 @@ namespace OASystem.API.OAMethodLib
             }
             }
 
 
             //计算时间段内工作日
             //计算时间段内工作日
-            int work_days = GetWorkDays(startDt, endDt);
+            int work_days = await GetWorkDays(thisYearMonth);
+            if (work_days <=0)
+            {
+                _result.Msg = thisYearMonth+" 工作日未设置,请前往《工作日管理页面》设置!";
+                return _result;
+            }
 
 
             UserIdListView userIdListView = await _qiYeWeChatApiService.GetUserIdListAsync();
             UserIdListView userIdListView = await _qiYeWeChatApiService.GetUserIdListAsync();
             if (userIdListView.errcode != 0)
             if (userIdListView.errcode != 0)
@@ -762,7 +768,8 @@ namespace OASystem.API.OAMethodLib
                 return _result;
                 return _result;
             }
             }
 
 
-            List<string> qyWhchatIdList = userIdListView.dept_user.Select(it => it.userid).ToList();
+            List<string> qyWhchatIdList = new List<string>();
+            qyWhchatIdList = userIdListView.dept_user.Select(it => it.userid).ToList();
             CheckInDayDataView checkInDayDataView = await _qiYeWeChatApiService.GetCheckInDayDataAsync(qyWhchatIdList, startDt, endDt);
             CheckInDayDataView checkInDayDataView = await _qiYeWeChatApiService.GetCheckInDayDataAsync(qyWhchatIdList, startDt, endDt);
             if (checkInDayDataView.errcode != 0)
             if (checkInDayDataView.errcode != 0)
             {
             {
@@ -770,7 +777,8 @@ namespace OASystem.API.OAMethodLib
                 return _result;
                 return _result;
             }
             }
             //筛选出工作日日报
             //筛选出工作日日报
-            List<Root> workday_userRoots = checkInDayDataView.datas.Where(it => it.base_info.day_type == 0).ToList(); //工作日日报
+            List<Root> workday_userRoots = checkInDayDataView.datas.Where(it => it.base_info.day_type == 0 && it.base_info.record_type == 1).ToList(); //工作日日报
+            workday_userRoots = workday_userRoots.OrderBy(it => it.base_info.date).ToList();
 
 
             //获取 请假类型 Sp_Detail.template_id
             //获取 请假类型 Sp_Detail.template_id
             string leave_template_id = "C4NzTJCh1onCUK915rRkvy7Fh5Vqz4YbiEV9jrBY1";
             string leave_template_id = "C4NzTJCh1onCUK915rRkvy7Fh5Vqz4YbiEV9jrBY1";
@@ -787,6 +795,7 @@ namespace OASystem.API.OAMethodLib
                 {
                 {
                     string itemName = userNames.Where(it => it.Id == pm_wsInfo.UserId).FirstOrDefault().CnName;
                     string itemName = userNames.Where(it => it.Id == pm_wsInfo.UserId).FirstOrDefault().CnName;
 
 
+
                     //补贴 金额
                     //补贴 金额
                     decimal meal_subsidy = 0.00M;  // 午餐(午餐10元/天)  补贴 * 计算方式:单日上午请假时长(小时)大于或者等于三小时 没有餐补
                     decimal meal_subsidy = 0.00M;  // 午餐(午餐10元/天)  补贴 * 计算方式:单日上午请假时长(小时)大于或者等于三小时 没有餐补
 
 
@@ -825,42 +834,480 @@ namespace OASystem.API.OAMethodLib
                     sickLeave_deduction = dailyWage - sickLeave_dailywage;
                     sickLeave_deduction = dailyWage - sickLeave_dailywage;
 
 
                     List<Ex_Items> ex_Items = new List<Ex_Items>();//假勤 And 打卡备注集合
                     List<Ex_Items> ex_Items = new List<Ex_Items>();//假勤 And 打卡备注集合
-
+                    Ex_Items ex_Items_dk = new Ex_Items() { Type = "打卡" };    //打卡
+                    Ex_Items ex_Items_jq = new Ex_Items() { Type = "假勤" };   //假勤
                     #endregion
                     #endregion
 
 
-                    List<Root> userRoots = workday_userRoots.Where(it => it.base_info.name == itemName && it.base_info.day_type == 0).ToList(); //工作日日报
+
+                    List<Root> userRoots = new List<Root>();
+
+                    if (itemName == "蔡雯")
+                    {
+                        userRoots = workday_userRoots.Where(it => it.base_info.name == "蔡蔡" || it.base_info.name == "蔡雯").ToList(); //工作日日报 1-固定上下班;
+                    }
+                    else
+                    {
+                        userRoots = workday_userRoots.Where(it => it.base_info.name == itemName).ToList(); //工作日日报 1-固定上下班;
+                    } 
+
+                    //userRoots = userRoots.Distinct().ToList();
+                    userRoots = userRoots.OrderBy(it => it.base_info.date).ToList();
+
 
 
                     int dk_work_days = userRoots.Count;   //应出勤天数 
                     int dk_work_days = userRoots.Count;   //应出勤天数 
 
 
+                    if (dk_work_days > work_days)
+                    {
+                        dk_work_days = work_days;
+                    }
                     meal_subsidy = dk_work_days * 10; //应发放餐补
                     meal_subsidy = dk_work_days * 10; //应发放餐补
 
 
 
 
                     if (!itemName.Equals("张海麟"))
                     if (!itemName.Equals("张海麟"))
                     {
                     {
+                        if (userRoots.Count <= 0)
+                        {
+                            _result.Msg = "【企业微信】【打卡】【获取打卡数据】【Msg】" + startDt + " - " + endDt + "打卡日数据 获取失败!";
+                            continue;
+                        }
+
+                        string acctid = userRoots[0].base_info.acctid;
+                        List<Ex_Item> ex_reissuecard_Items = new List<Ex_Item>(); //打卡类型 数据
+                        List<Sp_items> acc_sp_items = new List<Sp_items>(); //审批数据
+
+                        int user_probationary_bk_num = 0; 
+                                                          
+                                                          
+                        decimal user_probationary_bk_decimal = pm_wsInfo.Floats; //绩效工资为0 则为试用员工
+
+                        #region 迟到 早退 矿工
+
+                        int user_cd_zt_num = 0; //早退/迟到 次数 10分钟内 2次以内不记处罚 三次及以上50一次
                         foreach (var root in userRoots)
                         foreach (var root in userRoots)
                         {
                         {
                             List<Holiday_infos> holiday_Infos = root.holiday_infos; //当天假勤信息
                             List<Holiday_infos> holiday_Infos = root.holiday_infos; //当天假勤信息
                             List<Exception_infos> exception_infos = root.exception_infos; //当天校准状态信息
                             List<Exception_infos> exception_infos = root.exception_infos; //当天校准状态信息
                             List<Sp_items> sp_Items = root.sp_items;//当天假勤统计信息
                             List<Sp_items> sp_Items = root.sp_items;//当天假勤统计信息
-                            #region 假勤处理
                             if (sp_Items.Count > 0)
                             if (sp_Items.Count > 0)
                             {
                             {
+                                sp_Items = sp_Items.Where(it => it.count > 0).ToList();
+                                acc_sp_items.AddRange(sp_Items);
+                            }
+
+                            foreach (var exception_info in exception_infos)
+                            {
+                                decimal timelength = ConvertToDecimal((Convert.ToDecimal(exception_info.duration) / 3600.00M) * 60.00M);  //时长 分钟
+
+                                if (timelength == 9) timelength = 7.50M;
+
+                                int exception = exception_info.exception; //异常类型
+                                decimal day_miner_unit = ConvertToDecimal(dailyWage / 15);  //以0.5小时为单位 
+
+                                //1:一个自然月内,不足 10 分钟的迟到/早退,不超过 2 次的部分,不做处罚;3 次及以上,按50 元 / 次处罚;
+                                //2:超过 10 分钟(含 10 分钟),不足 60 分钟的迟到 / 早退,按 50 元 / 次处罚;
+                                //3:超过 60 分钟(含 60 分钟),不足 3 小时的迟到 / 早退,且无请假者,按旷工半日处理;超过 3 小时的迟到 / 早退,且无请假者,按旷工一日处理。
+
+                                long date = root.base_info.date; //当日工作日期
+                                long earliest_time = root.summary_info.earliest_time; //最早打卡时间
+                                long lastest_time = root.summary_info.lastest_time; //最晚打卡时间
+                                long this_date = date + earliest_time;
+                                DateTime thisDt = TimeZone.CurrentTimeZone.ToLocalTime(new DateTime(1970, 1, 1)).AddTicks(this_date * 10000000);
+
+                                string thisDtStr = thisDt.ToString("yyyy-MM-dd");
+
+                                Ex_Item beLate_belate_ex = new Ex_Item()
+                                {
+                                    SubTypeId = 4,
+                                    SubType = "旷工",
+                                    Duration = timelength,
+                                    StartTimeDt = Convert.ToDateTime(root.base_info.dateDt.ToString("yyyy-MM-dd HH:mm:ss")),
+                                    Unit = "分钟",
+                                };
+
+                                decimal day_deduction = 0.00M;
+                                //1 - 迟到;2 - 早退;3 - 缺卡;4 - 旷工;5 - 地点异常;6 - 设备异常
+                                if (exception == 1)   //迟到
+                                {
+                                    if (timelength < 10)
+                                    {
+                                        user_cd_zt_num++;
+                                        beLate_belate_ex.SubTypeId = 1;
+                                        beLate_belate_ex.SubType = "迟到";
+
+                                        string thisStartDt = (TimeZone.CurrentTimeZone.ToLocalTime(new DateTime(1970, 1, 1)).AddTicks((date + earliest_time) * 10000000))
+                                                              .ToString("yyyy-MM-dd HH:mm:ss");
+                                        beLate_belate_ex.StartTimeDt = Convert.ToDateTime(thisStartDt);
+
+                                        if (user_cd_zt_num >= 3)
+                                        {
+                                            day_deduction = 50.00M;
+                                        }
+                                        else
+                                        {
+                                            day_deduction = 0.00M;
+                                        }
+                                        beLate_deduction += day_deduction; //迟到扣款 总额
+                                    }
+                                    else if (timelength >= 10 && timelength <= 60)
+                                    {
+                                        string thisStartDt = (TimeZone.CurrentTimeZone.ToLocalTime(new DateTime(1970, 1, 1)).AddTicks((date + earliest_time) * 10000000))
+                                                              .ToString("yyyy-MM-dd HH:mm:ss");
+                                        beLate_belate_ex.StartTimeDt = Convert.ToDateTime(thisStartDt);
+
+                                        day_deduction = 50.00M;
+                                        beLate_deduction += day_deduction; //迟到扣款 总额
+                                        beLate_belate_ex.SubTypeId = 1;
+                                        beLate_belate_ex.SubType = "迟到";
+                                    }
+                                    else if (timelength > 60 && timelength <= 180)
+                                    {
+                                        day_deduction = day_miner_unit * 6; //3小时
+                                        meal_deduction += 10.00M;  //餐补扣款
+                                        absenteeism_deduction += day_deduction; //矿工半日
+                                    }
+                                    else
+                                    {
+                                        day_deduction = dailyWage;
+                                        absenteeism_deduction += day_deduction; //矿工一日
+                                        meal_deduction += 10.00M;
+                                    }
+
+                                    beLate_belate_ex.Deduction = day_deduction;
+                                    ex_reissuecard_Items.Add(beLate_belate_ex);
+                                }
+                                else if (exception == 2)  //早退
+                                {
+                                    if (timelength < 10)
+                                    {
+                                        user_cd_zt_num++;
+                                        beLate_belate_ex.SubTypeId = 2;
+                                        beLate_belate_ex.SubType = "早退";
+                                        string thisEndDt = TimeZone.CurrentTimeZone.ToLocalTime(new DateTime(1970, 1, 1)).AddTicks((date + lastest_time) * 10000000)
+                                                           .ToString("yyyy-MM-dd HH:mm:ss");
+                                        beLate_belate_ex.StartTimeDt = Convert.ToDateTime(thisEndDt);
+
+
+                                        if (user_cd_zt_num >= 3)
+                                        {
+                                            day_deduction = 50.00M;
+                                        }
+                                        else
+                                        {
+                                            day_deduction = 0.00M;
+                                        }
+                                        early_deduction += day_deduction;  //早退扣款 总计
+                                    }
+                                    else if (timelength >= 10 && timelength <= 60)
+                                    {
+                                        day_deduction = 50.00M;
+                                        early_deduction += day_deduction;  //早退扣款 总计
+                                        beLate_belate_ex.SubTypeId = 2;
+                                        beLate_belate_ex.SubType = "早退";
+                                        string thisEndDt = TimeZone.CurrentTimeZone.ToLocalTime(new DateTime(1970, 1, 1)).AddTicks((date + lastest_time) * 10000000)
+                                                           .ToString("yyyy-MM-dd HH:mm:ss");
+                                        beLate_belate_ex.StartTimeDt = Convert.ToDateTime(thisEndDt);
+                                    }
+                                    else if (timelength > 60 && timelength <= 180)
+                                    {
+                                        day_deduction = day_miner_unit * 6; //3小时
+                                        absenteeism_deduction += day_deduction; //矿工半日
+
+                                    }
+                                    else
+                                    {
+                                        day_deduction = dailyWage;
+                                        meal_deduction += 10.00M;
+                                        absenteeism_deduction += day_deduction; //矿工一日
+                                    }
+
+                                    beLate_belate_ex.Deduction = day_deduction;
+                                    ex_reissuecard_Items.Add(beLate_belate_ex);
+                                }
+                                else if (exception == 3) //缺卡
+                                {
+                                    if (root.exception_infos.Count == 2)
+                                    {
+                                        day_deduction = dailyWage;
+                                        meal_deduction += 10.00M;
+                                        absenteeism_deduction += day_deduction;
+                                        beLate_belate_ex.Reason = "上午-下午 缺卡/未打卡为旷工 一天";
+                                        beLate_belate_ex.Deduction = day_deduction;
+                                        ex_reissuecard_Items.Add(beLate_belate_ex);
+
+                                    }
+                                    else if (root.exception_infos.Count == 1)
+                                    {
+
+                                        if (earliest_time == lastest_time)
+                                        {
+                                            DateTime thisDt1 = Convert.ToDateTime(thisDtStr + " 12:00");
+
+
+                                            if (thisDt <= thisDt1)  //旷工 下午
+                                            {
+                                                day_deduction = day_miner_unit * 9; //4.5小时
+                                                meal_deduction += 10.00M;
+                                                absenteeism_deduction += day_deduction; //矿工半日
+                                                beLate_belate_ex.Reason = thisDtStr + " 下午(18:00)缺卡/未打卡视为下午旷工(4.5小时)";
+
+                                                beLate_belate_ex.Deduction = day_deduction;
+                                                ex_reissuecard_Items.Add(beLate_belate_ex);
+                                            }
+                                            else if (thisDt >= thisDt1) //旷工下午
+                                            {
+                                                day_deduction = day_miner_unit * 6; //3小时
+                                                absenteeism_deduction += day_deduction; //矿工半日
+                                                beLate_belate_ex.Reason = thisDtStr + " 上午(09:00)缺卡/未打卡视为上午旷工(3小时)";
+
+                                                beLate_belate_ex.Deduction = day_deduction;
+                                                ex_reissuecard_Items.Add(beLate_belate_ex);
+                                            }
+                                            else //矿工一日
+                                            {
+
+                                                day_deduction = dailyWage;
+                                                meal_deduction += 10.00M;
+                                                absenteeism_deduction += day_deduction;
+                                                beLate_belate_ex.Reason = thisDtStr + " 上午(09:00)-下午(18:00) 缺卡/未打视为旷工一天(7.5小时)";
+
+                                                beLate_belate_ex.Deduction = day_deduction;
+                                                ex_reissuecard_Items.Add(beLate_belate_ex);
+                                            }
+
+
+                                        }
+                                    }
+                                }
+                                else if (exception == 4) //旷工
+                                {
+                                    if (timelength > 60 && timelength <= 180)
+                                    {
+                                        day_deduction = day_miner_unit * 6; //3小时
+                                        beLate_belate_ex.Reason = thisDtStr + " 缺卡/未打视为旷工上午(3小时)";
+                                        meal_deduction += 10.00M;
+                                        absenteeism_deduction += day_deduction; //矿工半日
+
+                                    }
+                                    else
+                                    {
+                                        day_deduction = dailyWage;
+                                        beLate_belate_ex.Reason = thisDtStr + " 上午(09:00)-下午(18:00) 缺卡/未打视为旷工一天(7.5小时)";
+                                        meal_deduction += 10.00M;
+                                        absenteeism_deduction += day_deduction; //矿工一日
+                                    }
+
+                                    beLate_belate_ex.Deduction = day_deduction;
+                                    ex_reissuecard_Items.Add(beLate_belate_ex);
+                                }
 
 
                             }
                             }
+                        }
 
 
+                        #endregion
 
 
-                            #endregion
+                        #region 假勤/补卡次数 审批
 
 
-                            #region 打卡补卡
+                        int leaveNum = 0; //请假次数
+                        int reissuecardNum = 0; //补卡次数\
+                        //类型:1 - 请假;2 - 补卡;3 - 出差;4 - 外出;100 - 外勤
+                        leaveNum = acc_sp_items.Where(it => it.type == 1).ToList().Count();
+                        reissuecardNum = acc_sp_items.Where(it => it.type == 2).ToList().Count();
 
 
-                            #endregion
+                        //请假审批
+                        if (leaveNum > 0)
+                        {
+                            List<Sp_Detail> sp_leave_details = new List<Sp_Detail>();
+                            sp_leave_details = await _qiYeWeChatApiService.GetApprovalDetailsAsync(startDt, endDt, acctid, 2, 1); //时间段内所有 已同意的 请假 审批数据
+                            if (sp_leave_details.Count <= 0)
+                            {
+                                _result.Msg += startDt + " - " + endDt + "  " + itemName + "  请假 审批数据获取未获取到!\r\n";
+                                //continue;
+                            }
+                            List<Ex_Item> ex_ItemInfos = new List<Ex_Item>();
 
 
-                            #region 迟到 早退 矿工
+                            foreach (Sp_Detail sp_item in sp_leave_details)
+                            {
+                                Apply_data? apply_data = sp_item.apply_data;
+                                if (apply_data != null)
+                                {
+                                    List<ContentsItem> contents = apply_data.contents;
+                                    ContentsItem content_Vacation = contents.Where(it => it.control == "Vacation").FirstOrDefault();  //请假类型 
+                                    ContentsItem content_Textarea = contents.Where(it => it.control == "Textarea").FirstOrDefault(); //多行文本 
 
 
+                                    if (content_Vacation != null)
+                                    {
+                                        Vacation vacation = content_Vacation.value.vacation;
+                                        Attendance attendance = vacation.attendance; //假勤组件
+                                        Selector selector = vacation.selector;       //请假类型
 
 
-                            #endregion
+                                        List<OptionsItem> optionsItems = selector.options; //key 请假类型 id
+                                        List<TitleItem> value = optionsItems[0].value; // value 文本描述值
+
+                                        int leaveType = int.Parse(optionsItems[0].key);  //key 请假子类型 id
+                                        Date_range date_Range = attendance.date_range;
+
+                                        //筛选 不在工作日内的假勤申请
+                                        if (startDt >= date_Range.new_begin_dt || date_Range.new_end_dt > endDt)
+                                        {
+                                            continue;
+                                        }
+                                       
+
+                                        decimal thisTypeDeduction = 0.00M;//当前类型扣款
+                                        string leave_starttime = date_Range.new_begin_dt.ToString("HH:mm");
+                                        string leave_endtime = date_Range.new_end_dt.ToString("HH:mm");
+
+                                        string typeName = string.Empty;
+                                        string unit = string.Empty;
+                                        int leaveTypeId = leaveType;
+                                        var leaveTypeData = vacationLeaveTypes.Where(it => it.id == leaveTypeId).FirstOrDefault();
+
+                                        if (leaveTypeData != null) { typeName = leaveTypeData.name; }
+
+                                        string startTime = string.Empty;
+                                        string endTime = string.Empty;
+                                        //计算请假类型扣款金额
+                                        decimal new_duration = 0.00M;
+                                        if (date_Range.type == "halfday")
+                                        {
+                                            new_duration = Convert.ToDecimal(date_Range.new_duration) / 86400M;
+                                            unit = "天";
+                                            startTime = date_Range.new_begin_dt.ToString("yyyy-MM-dd") + " 09:00";
+                                            endTime = date_Range.new_begin_dt.ToString("yyyy-MM-dd") + " 18:00";
+                                        }
+                                        else if (date_Range.type == "hour")
+                                        {
+                                            new_duration = Convert.ToDecimal(date_Range.new_duration) / 3600M;
+                                            unit = "小时";
+                                            startTime = date_Range.new_begin_dt.ToString("yyyy-MM-dd HH:mm:ss");
+                                            endTime = date_Range.new_end_dt.ToString("yyyy-MM-dd HH:mm:ss");
+                                        }
+                                        decimal leave_meals = 0.00M;
+
+                                        //计算餐补 假勤类型扣款
+                                        CalculateTypeFee(leaveType, date_Range.type, leave_starttime, leave_endtime, amountPayable, work_days, new_duration,
+                                            out leave_meals, out thisTypeDeduction);
+
+                                        #region 累计类型扣款
+                                        //1年假;2事假;3病假;4调休假;5婚假;6产假;7陪产假;8其他;9丧假
+
+                                        if (leaveType == 2) //事假
+                                        {
+                                            personalLeaveTotal += thisTypeDeduction;
+                                        }
+                                        else if (leaveType == 3) //病假
+                                        {
+                                            sickLeaveTotal = thisTypeDeduction;
+                                        }
+
+                                        #endregion
+
+                                        meal_deduction += leave_meals;
+
+
+                                        Ex_Item ex_Item = new Ex_Item()
+                                        {
+                                            SubTypeId = leaveType,
+                                            SubType = typeName,
+                                            StartTimeDt = Convert.ToDateTime(startTime),
+                                            EndTimeDt = Convert.ToDateTime(endTime),
+                                            Duration = new_duration,
+                                            Unit = unit,
+                                            Deduction = thisTypeDeduction,
+                                            //Reason = apply_data.reason,
+                                            Apply_time_dt = Convert.ToDateTime(sp_item.apply_time_dt.ToString("yyyy-MM-dd HH:mm:ss")),
+                                            //Approval_name = sp_item.approval_name,
+                                        };
+                                        ex_ItemInfos.Add(ex_Item);
+                                    }
+                                }
+                            }
+
+                            if (ex_ItemInfos.Count > 0)
+                            {
+                                ex_Items_jq.Ex_ItemInfo = ex_ItemInfos.OrderBy(it => it.StartTimeDt).ThenBy(it => it.Apply_time_dt).ToList();
+                                ex_Items.Add(ex_Items_jq);
+                            }
+
+                        }
+
+                        //打卡补卡
+                        //补卡:员工发现自己漏打卡时,需及时提起补卡申请,并说明情况。
+                        //试用期员工每月有 2 次补卡机会,超过 2 次不足 5 次的部分,按 10 元/次处罚,5 次及以上的漏卡,按 50 元/次处罚;
+                        //正式员工每月 3 次以内的补卡,按 10 元/次处罚,3 次及以上的漏卡,按 50 元/次处罚。
+                        if (reissuecardNum > 0)
+                        {
+                            List<Sp_Detail> sp_buka_details = new List<Sp_Detail>();
+                            sp_buka_details = await _qiYeWeChatApiService.GetApprovalDetailsAsync(startDt, endDt, acctid, 2, 2); //时间段内所有 已同意的 请假 审批数据
+
+                            int bukaNum = 1;
+                            foreach (var item in sp_buka_details)
+                            {
+                                Apply_data? apply_data = item.apply_data;
+                                if (apply_data != null)
+                                {
+                                    List<ContentsItem> contents = apply_data.contents;
+                                    ContentsItem content_Vacation = contents.Where(it => it.control == "PunchCorrection").FirstOrDefault();  //请假类型 
+                                    ContentsItem content_Textarea = contents.Where(it => it.control == "Textarea").FirstOrDefault(); //多行文本 
+
+                                    if (content_Vacation != null)
+                                    {
+                                        var punch_correction = content_Vacation.value.punch_correction;
+
+                                        DateTime bukaDt = punch_correction.time_dt;
+
+                                        //筛选 不在工作日内的假勤申请
+                                        if (startDt >= bukaDt || bukaDt > endDt)
+                                        {
+                                            continue;
+                                        }
+                                        
+                                        decimal bukaPrice = 0.00M;
+
+                                        if (user_probationary_bk_decimal == 0) //计算试用员工补卡次数
+                                        {
+                                            if (bukaNum <= 2) bukaPrice = 0.00M;
+                                            else if (bukaNum <= 4 && bukaNum > 2) bukaPrice = 10.00M;
+                                            else bukaPrice = 50.00M;
+
+                                        } 
+                                        else   //计算正式员工补卡次数
+                                        {
+                                            if (bukaNum <= 2) bukaPrice = 10.00M;
+                                            else bukaPrice = 50.00M;
+                                        }
+
+                                        
+                                        var app_data = item.apply_data;
+                                        var punch_correction1 = app_data.contents[0].value.punch_correction; //未打卡时间
+                                        var punch_correction2 = app_data.contents[1].value;
+
+                                        Ex_Item ex_reissueCard = new Ex_Item()
+                                        {
+                                            SubTypeId = 7,
+                                            SubType = "打卡补卡",
+                                            StartTimeDt = Convert.ToDateTime(punch_correction1.time_dt.ToString("yyyy-MM-dd HH:mm:ss")),  //未打卡时间
+                                            Deduction = bukaPrice,
+                                            Reason = punch_correction2.text,
+                                            Unit = string.Empty
+                                        };
+                                        unprinted_deduction += bukaPrice;
+                                        ex_reissuecard_Items.Add(ex_reissueCard);
+                                        bukaNum++;
+                                    }
+                                }
+                            }
+                        }
+
+                        if (ex_reissuecard_Items.Count > 0)
+                        {
+                            ex_Items_dk.Ex_ItemInfo = ex_reissuecard_Items;
+                            ex_Items_dk.Ex_ItemInfo = ex_reissuecard_Items.OrderBy(it => it.SubTypeId).ThenBy(it => it.StartTimeDt).ToList();
+                            ex_Items.Add(ex_Items_dk);
                         }
                         }
 
 
+                        #endregion
+
                     }
                     }
                     else
                     else
                     {
                     {
@@ -878,7 +1325,15 @@ namespace OASystem.API.OAMethodLib
                     }
                     }
                     else
                     else
                     {
                     {
-                        salaryTotal = (dk_work_days * dailyWage) + mealTotal;    //应发合计
+                        if (itemName.Equals("张海麟"))
+                        {
+                            salaryTotal = amountPayable + mealTotal;    //应发合计
+                        }
+                        else
+                        {
+                            salaryTotal = (dk_work_days * dailyWage) + mealTotal;    //应发合计
+                        }
+                            
                     }
                     }
 
 
 
 
@@ -892,7 +1347,7 @@ namespace OASystem.API.OAMethodLib
                     #region 处理当月工资数据
                     #region 处理当月工资数据
                     pm_wsInfo.YearMonth = thisYearMonth;
                     pm_wsInfo.YearMonth = thisYearMonth;
                     pm_wsInfo.StartDate = startDt.ToString("yyyy-MM-dd");
                     pm_wsInfo.StartDate = startDt.ToString("yyyy-MM-dd");
-                    pm_wsInfo.EndDate = endDt.AddDays(-1).ToString("yyyy-MM-dd");
+                    pm_wsInfo.EndDate = endDt.ToString("yyyy-MM-dd");
                     pm_wsInfo.WorkDays = work_days;        //当月应出勤天数
                     pm_wsInfo.WorkDays = work_days;        //当月应出勤天数
                     pm_wsInfo.RegularDays = dk_work_days;  //当月正常出勤天数
                     pm_wsInfo.RegularDays = dk_work_days;  //当月正常出勤天数
 
 
@@ -1005,8 +1460,6 @@ namespace OASystem.API.OAMethodLib
             string am_starttime = "08:59";
             string am_starttime = "08:59";
             string am_endtime = "13:01";
             string am_endtime = "13:01";
 
 
-            
-
             decimal personalkLeave_dailywage_day = ConvertToDecimal( amountPayable / work_days); //日薪 = 事假日薪 *计算方式:日平均工资 = 当月应发工资 /当月应出勤天数。
             decimal personalkLeave_dailywage_day = ConvertToDecimal( amountPayable / work_days); //日薪 = 事假日薪 *计算方式:日平均工资 = 当月应发工资 /当月应出勤天数。
 
 
             //半小时单位
             //半小时单位
@@ -1030,17 +1483,22 @@ namespace OASystem.API.OAMethodLib
                     {
                     {
                         decimal leave_halfHour = Convert.ToDecimal(duration) / Convert.ToDecimal(0.5);
                         decimal leave_halfHour = Convert.ToDecimal(duration) / Convert.ToDecimal(0.5);
                         typeDeduction = personalkLeave_dailywage_halfhour * leave_halfHour;
                         typeDeduction = personalkLeave_dailywage_halfhour * leave_halfHour;
-                        if (duration >= 3 && duration <= 7.5M) //单天请假三小时 && 请假时间在上午 则没有餐补
+                        if (duration >= 3 && duration <= 7.5M) //单天请假三小时
                         {
                         {
+                            mealDeduction = 10; //餐补扣款
+
+                            #region 上午请假超三小时 扣款
                             //处理开始时间
                             //处理开始时间
-                            if (startTime.CompareTo(am_starttime) > 0 && startTime.CompareTo(am_endtime) < 0)
-                            {
-                                //处理结束时间
-                                //if (endTime.CompareTo(am_starttime) > 0 && endTime.CompareTo(am_endtime) > 0)
-                                //{
-                                    mealDeduction = 10; //餐补扣款
-                                //}
-                            }
+                            //if (startTime.CompareTo(am_starttime) > 0 && startTime.CompareTo(am_endtime) < 0)
+                            //{
+                            //    //处理结束时间
+                            //    if (endTime.CompareTo(am_starttime) > 0 && endTime.CompareTo(am_endtime) > 0)
+                            //    {
+                            //        mealDeduction = 10; //餐补扣款
+                            //    }
+                            //}
+                            #endregion
+
                         }
                         }
                         else if (duration > 7.5M) //多天计算
                         else if (duration > 7.5M) //多天计算
                         {
                         {
@@ -1088,15 +1546,20 @@ namespace OASystem.API.OAMethodLib
                         typeDeduction = sickLeave_dailywage_halfhour_deduction *sickLeave_halfHour ;
                         typeDeduction = sickLeave_dailywage_halfhour_deduction *sickLeave_halfHour ;
                         if (duration >= 3 && duration <= 7.5M) //单天请假三小时 && 请假时间在上午 则没有餐补
                         if (duration >= 3 && duration <= 7.5M) //单天请假三小时 && 请假时间在上午 则没有餐补
                         {
                         {
-                            //处理开始时间
-                            if (startTime.CompareTo(am_starttime) > 0 && startTime.CompareTo(am_endtime) < 0)
-                            {
-                                //处理结束时间
-                                if (endTime.CompareTo(am_starttime) > 0 && endTime.CompareTo(am_endtime) < 0)
-                                {
-                                    mealDeduction = 10; //餐补扣款
-                                }
-                            }
+                            mealDeduction = 10; //餐补扣款
+
+                            #region 上午请假超三小时扣款
+                            ////处理开始时间
+                            //if (startTime.CompareTo(am_starttime) > 0 && startTime.CompareTo(am_endtime) < 0)
+                            //{
+                            //    //处理结束时间
+                            //    if (endTime.CompareTo(am_starttime) > 0 && endTime.CompareTo(am_endtime) < 0)
+                            //    {
+                            //        mealDeduction = 10; //餐补扣款
+                            //    }
+                            //}
+                            #endregion
+
                         }
                         }
                         else if (duration > 7.5M) //多天计算
                         else if (duration > 7.5M) //多天计算
                         {
                         {
@@ -1162,8 +1625,8 @@ namespace OASystem.API.OAMethodLib
         public static void CalculateTypeFeeSub(string date_Range_type, string startTime, string endTime, decimal duration, out decimal mealDeduction)
         public static void CalculateTypeFeeSub(string date_Range_type, string startTime, string endTime, decimal duration, out decimal mealDeduction)
         {
         {
             mealDeduction = 0;
             mealDeduction = 0;
-            string am_starttime = "08:59";
-            string am_endtime = "13:01";
+            string am_starttime = "09:00";
+            string am_endtime = "11:59";
 
 
             if (date_Range_type == "halfday")
             if (date_Range_type == "halfday")
             {
             {
@@ -1173,15 +1636,28 @@ namespace OASystem.API.OAMethodLib
             {
             {
                 if (duration >= 3 && duration < 7) //单天请假三小时 && 请假时间在上午 则没有餐补
                 if (duration >= 3 && duration < 7) //单天请假三小时 && 请假时间在上午 则没有餐补
                 {
                 {
-                    //处理开始时间
-                    if (startTime.CompareTo(am_starttime) > 0 && startTime.CompareTo(am_endtime) < 0)
-                    {
-                        //处理结束时间
-                        if (endTime.CompareTo(am_starttime) > 0 && endTime.CompareTo(am_endtime) < 0)
-                        {
-                            mealDeduction = 10; //餐补扣款
-                        }
-                    }
+                    mealDeduction = 10; //餐补扣款
+
+                    #region 上午请假超三小时 扣款
+
+                    //TimeSpan start_ts = TimeSpan.Parse(startTime); // 10 PM
+                    //TimeSpan end_ts = TimeSpan.Parse(endTime);     // 2  AM
+
+                    //TimeSpan am_start_ts = TimeSpan.Parse(am_starttime); // 10 PM
+                    //TimeSpan am_end_ts = TimeSpan.Parse(am_endtime);     // 2  AM
+
+                    ////处理开始时间
+                    //if (start_ts >= am_start_ts && start_ts <= am_end_ts)
+                    //{
+                    //    //处理结束时间
+                    //    if (end_ts > am_start_ts && end_ts >= am_end_ts)
+                    //    {
+                    //        mealDeduction = 10; //餐补扣款
+                    //    }
+                    //}
+
+                    #endregion
+
                 }
                 }
                 else if (duration >= 7 && duration <= 7.50M )
                 else if (duration >= 7 && duration <= 7.50M )
                 {
                 {
@@ -1203,11 +1679,13 @@ namespace OASystem.API.OAMethodLib
                         int lastHours = (Convert.ToDateTime(endTime) - Convert.ToDateTime("09:00")).Hours;
                         int lastHours = (Convert.ToDateTime(endTime) - Convert.ToDateTime("09:00")).Hours;
                         if (lastHours >= 3)
                         if (lastHours >= 3)
                         {
                         {
-                            //处理结束时间
-                            if (endTime.CompareTo(am_starttime) > 0 && endTime.CompareTo(am_endtime) < 0)
-                            {
-                                mealDeduction += 10; //餐补扣款
-                            }
+                            mealDeduction += 10; //餐补扣款
+
+                            ////处理结束时间
+                            //if (endTime.CompareTo(am_starttime) > 0 && endTime.CompareTo(am_endtime) < 0)
+                            //{
+                            //    mealDeduction += 10; //餐补扣款
+                            //}
                         }
                         }
                     }
                     }
                 }
                 }
@@ -1291,77 +1769,20 @@ namespace OASystem.API.OAMethodLib
         /// <param name="startDt"></param>
         /// <param name="startDt"></param>
         /// <param name="endDt"></param>
         /// <param name="endDt"></param>
         /// <returns></returns>
         /// <returns></returns>
-        public static int GetWorkDays(DateTime startDt,DateTime endDt) 
+        public static async Task<int> GetWorkDays(string yearMonth) 
         {
         {
-            string startYear = startDt.ToString("yyyy");
-            string endYear = endDt.ToString("yyyy");
-
-            var holidays = AppSettingsHelper.Get<HoliDaysDataSourceView>("HoliDayDataSource");
-            List<HoliDaysDataSourceView> holiDatas = new List<HoliDaysDataSourceView>();
-            if (startYear.Equals(endYear))
-            {
-                holiDatas.Add(holidays.Where(it => it.Year == startYear).FirstOrDefault());
-            }
-            else
-            {
-                holiDatas.Add(holidays.Where(it => it.Year == startYear).FirstOrDefault());
-                holiDatas.Add(holidays.Where(it => it.Year == endYear).FirstOrDefault());
-            }
+            int workdays = 0;
+            string sql = string.Format(@"Select * From  Pm_WageIssueWorkingDay 
+                                         Where Isdel = 0 And YearMonth = '{0}' ", yearMonth);
 
 
-            int _totalDays = (int)(endDt - startDt).TotalDays;
-            List<HolidaysInfo> _holidays = new List<HolidaysInfo>();
-            List<HolidaysInfo> _weekdys = new List<HolidaysInfo>();
-            List<HolidaysInfo> _workdays = new List<HolidaysInfo>();
-            string[] Day = new string[] { "星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六" };
+            var data = await _usersRep._sqlSugar.SqlQueryable<WageYearMonthView>(sql).FirstAsync();
 
 
-            for (int i = 0; i <= _totalDays; i++)
+            if (data != null)
             {
             {
-                string thisYear = startDt.ToString("yyyy");
-                string thisDt = startDt.ToString("MM-dd");
-                string thisYMD = startDt.ToString("yyyy-MM-dd");
-                string thisWeek = Day[Convert.ToInt32(startDt.DayOfWeek.ToString("d"))].ToString();
-
-
-                HoliDaysDataSourceView holiDaysData = new HoliDaysDataSourceView();
-                holiDaysData = holiDatas.Where(it => it.Year == thisYear).FirstOrDefault();
-                //周末
-                if (thisWeek.Equals("星期六") || thisWeek.Equals("星期日"))
-                {
-                    HolidaysInfo holidaysInfo = new HolidaysInfo();
-                    holidaysInfo = holiDaysData.Holidays.Where(it => it.HoliDate == thisDt).FirstOrDefault();
-                    if (holidaysInfo != null)
-                    {
-                        if (holidaysInfo.HoliName.Equals("工作日"))
-                        {
-                            _workdays.Add(new HolidaysInfo { HoliDate = thisYMD, HoliName = Day[Convert.ToInt32(startDt.DayOfWeek.ToString("d"))].ToString() });
-                            startDt = startDt.AddDays(1);
-                            continue;
-                        }
-                    }
-                    _weekdys.Add(new HolidaysInfo { HoliDate = thisYMD, HoliName = thisWeek });
-                    startDt = startDt.AddDays(1);
-                    continue;
-                }
-                
-                //节假日
-                if (holiDaysData != null)
-                {
-                    HolidaysInfo holidaysInfo = new HolidaysInfo();
-                    holidaysInfo = holiDaysData.Holidays.Where(it => it.HoliDate == thisDt).FirstOrDefault();
-                    if (holidaysInfo != null)
-                    {
-                        _holidays.Add(new HolidaysInfo { HoliDate = thisYMD, HoliName = holidaysInfo.HoliName });
-                        startDt = startDt.AddDays(1);
-                        continue;
-                    }
-                }
-                //工作日
-                _workdays.Add(new HolidaysInfo { HoliDate = thisYMD, HoliName = Day[Convert.ToInt32(startDt.DayOfWeek.ToString("d"))].ToString() });
-
-                startDt = startDt.AddDays(1);
+                workdays = data.Workdays;
             }
             }
 
 
-            return _workdays.Count();
+            return workdays;
         }
         }
     }
     }
 }
 }

+ 323 - 26
OASystem/OASystem.Api/OAMethodLib/QiYeWeChatAPI/QiYeWeChatApiService.cs

@@ -1,4 +1,5 @@
-using OASystem.Domain.Dtos.QiYeWeChat;
+using Microsoft.AspNetCore.Identity;
+using OASystem.Domain.Dtos.QiYeWeChat;
 using OASystem.Domain.ViewModels.JuHeExchangeRate;
 using OASystem.Domain.ViewModels.JuHeExchangeRate;
 using OASystem.Domain.ViewModels.QiYeWeChat;
 using OASystem.Domain.ViewModels.QiYeWeChat;
 using Org.BouncyCastle.Ocsp;
 using Org.BouncyCastle.Ocsp;
@@ -298,7 +299,7 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
         /// <param name="startDt"></param>
         /// <param name="startDt"></param>
         /// <param name="endDt"></param>
         /// <param name="endDt"></param>
         /// <returns></returns>
         /// <returns></returns>
-        public async Task<CheckInView> GetCheckin_MonthDataAsync(DateTime startDt,DateTime endDt)
+        public async Task<CheckInView> GetCheckin_MonthDataAsync1(DateTime startDt,DateTime endDt)
         {
         {
             CheckInView checkInView = new CheckInView();
             CheckInView checkInView = new CheckInView();
 
 
@@ -347,6 +348,152 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
             return checkInView;
             return checkInView;
         }
         }
 
 
+        /// <summary>
+        /// 获取月打卡记录数据
+        /// </summary>
+        /// <param name="startDt"></param>
+        /// <param name="endDt"></param>
+        /// <returns></returns>
+        public async Task<CheckInView> GetCheckin_MonthDataAsync(DateTime startDt, DateTime endDt)
+        {
+            CheckInView checkInView = new CheckInView();
+
+            //获取员工Id
+            UserIdListView userIdListView = await GetUserIdListAsync();
+            if (userIdListView.errcode != 0)
+            {
+                checkInView.errcode = userIdListView.errcode;
+                checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】{0}", userIdListView.errmsg);
+                return checkInView;
+            }
+
+            if (userIdListView.dept_user == null || userIdListView.dept_user.Count <= 0)
+            {
+                checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】未查出员工Id");
+                return checkInView;
+            }
+
+            //获取打卡数据 token
+            Access_TokenView access_Token = await GetTokenAsync(2);
+            if (access_Token.errcode != 0)
+            {
+                checkInView.errcode = access_Token.errcode;
+                checkInView.errmsg = string.Format("【企业微信】【获取月打卡数据】【Token】【Msg】{0}", access_Token.errmsg);
+                return checkInView;
+            }
+
+
+            #region 处理超31天的月报表数据
+
+            string url = string.Format("/cgi-bin/checkin/getcheckin_monthdata?access_token={0}", access_Token.access_token);
+
+            Checkin_MonthData_Request checkInReq = new Checkin_MonthData_Request();
+            checkInReq.access_token = access_Token.access_token;
+            checkInReq.useridlist = userIdListView.dept_user.Select(it => it.userid).ToList();
+
+            uint starttimeOne, endtimeOne, starttimeTwo, endtimeTwo;
+            int days = (int)(endDt - startDt).TotalDays;
+
+            if (days > 32) 
+            {
+                starttimeOne = (uint)((startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
+                endtimeOne = (uint)((startDt.AddDays(31).ToUniversalTime().Ticks - 621355968000000000) / 10000000);
+                starttimeTwo = (uint)((startDt.AddDays(32).ToUniversalTime().Ticks - 621355968000000000) / 10000000);
+                endtimeTwo = (uint)((endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
+
+                checkInReq.starttime = starttimeOne;
+                checkInReq.endtime = endtimeOne;
+                CheckInView checkInViewOne = new CheckInView();
+
+                var jsonOne = System.Text.Json.JsonSerializer.Serialize(checkInReq);
+                var contentOne = new StringContent(jsonOne, Encoding.UTF8, "application/json");
+                var create_ReqOne = await _httpClient.PostAsync(url, contentOne);
+                var stringResponseOne = await create_ReqOne.Content.ReadAsStringAsync();
+
+                checkInViewOne = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponseOne,
+                    new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+
+                if (checkInViewOne.errcode != 0)
+                {
+                    return checkInViewOne;
+                }
+
+                checkInReq.starttime = starttimeTwo;
+                checkInReq.endtime = endtimeTwo;
+                CheckInView checkInViewTwo = new CheckInView();
+
+
+
+                var jsonTwo = System.Text.Json.JsonSerializer.Serialize(checkInReq);
+                var contentTwo = new StringContent(jsonTwo, Encoding.UTF8, "application/json");
+                var create_ReqTwo = await _httpClient.PostAsync(url, contentTwo);
+                var stringResponseTwo = await create_ReqTwo.Content.ReadAsStringAsync();
+
+                checkInViewTwo = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponseTwo,
+                    new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+
+                if (checkInViewTwo.errcode != 0)
+                {
+                    return checkInViewTwo;
+                }
+
+                #region 整合月报表数据 
+
+                foreach (var item_one in checkInViewOne.datas)
+                {
+                    foreach (var item_two in checkInViewTwo.datas)
+                    {
+                        if (item_one.base_info.acctid.Equals(item_two.base_info.acctid))
+                        {
+                            #region summary_info 汇总信息整合
+                            item_one.summary_info.work_days += item_two.summary_info.work_days; //应打卡天数
+                            item_one.summary_info.regular_days += item_two.summary_info.regular_days;  //正常天数
+                            item_one.summary_info.except_days += item_two.summary_info.except_days;   //异常天数
+                            item_one.summary_info.regular_work_sec += item_two.summary_info.regular_work_sec;  //实际工作时长,为统计周期每日实际工作时长之和
+                            item_one.summary_info.standard_work_sec += item_two.summary_info.standard_work_sec; //标准工作时长,为统计周期每日标准工作时长之和
+                            #endregion
+
+                            #region exception_infos 异常状态统计信息 整合
+                            item_one.exception_infos.AddRange(item_two.exception_infos);
+                            #endregion
+
+                            #region sp_items 	假勤统计信息整合
+                            item_one.sp_items.AddRange(item_two.sp_items);
+                            #endregion
+
+                            #region overwork_info 加班情况 整合
+                            item_one.overwork_info.workday_over_sec += item_two.overwork_info.workday_over_sec;   //工作日加班时长
+                            item_one.overwork_info.holidays_over_sec += item_two.overwork_info.holidays_over_sec;   //节假日加班时长
+                            item_one.overwork_info.restdays_over_sec += item_two.overwork_info.restdays_over_sec;   //休息日加班时长
+                            #endregion
+                            break;
+                        }
+                    }
+                }
+                #endregion
+                checkInView = checkInViewOne;
+            }
+            else
+            {
+                checkInReq.starttime = (uint)((startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
+                checkInReq.endtime = (uint)((endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
+
+                var json = System.Text.Json.JsonSerializer.Serialize(checkInReq);
+                var content = new StringContent(json, Encoding.UTF8, "application/json");
+                var create_Req = await _httpClient.PostAsync(url, content);
+                var stringResponse = await create_Req.Content.ReadAsStringAsync();
+
+                checkInView = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponse,
+                    new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+            }
+
+
+            #endregion
+
+            return checkInView;
+        }
+
+
         /// <summary>
         /// <summary>
         /// 获取月打卡数据(redis缓存)
         /// 获取月打卡数据(redis缓存)
         /// </summary>
         /// </summary>
@@ -510,7 +657,6 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
             return checkInDataView;
             return checkInDataView;
         }
         }
 
 
-
         /// <summary>
         /// <summary>
         /// 获取打卡日报数据
         /// 获取打卡日报数据
         /// </summary>
         /// </summary>
@@ -518,7 +664,7 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
         /// <param name="startDt"></param>
         /// <param name="startDt"></param>
         /// <param name="endDt"></param>
         /// <param name="endDt"></param>
         /// <returns></returns>
         /// <returns></returns>
-        public async Task<CheckInDayDataView> GetCheckInDayDataAsync(List<string> useridlist, DateTime startDt, DateTime endDt)
+        public async Task<CheckInDayDataView> GetCheckInDayDataAsync1(List<string> useridlist, DateTime startDt, DateTime endDt)
         {
         {
             CheckInDayDataView checkInDayDataView = new CheckInDayDataView();
             CheckInDayDataView checkInDayDataView = new CheckInDayDataView();
 
 
@@ -562,6 +708,115 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
 
 
             return checkInDayDataView;
             return checkInDayDataView;
         }
         }
+
+
+        /// <summary>
+        /// 获取打卡日报数据
+        /// </summary>
+        /// <param name="useridlist"></param>
+        /// <param name="startDt"></param>
+        /// <param name="endDt"></param>
+        /// <returns></returns>
+        public async Task<CheckInDayDataView> GetCheckInDayDataAsync(List<string> useridlist, DateTime startDt, DateTime endDt)
+        {
+            CheckInDayDataView checkInDayDataView = new CheckInDayDataView();
+
+            //参数处理
+            if (useridlist.Count <= 0)
+            {
+                checkInDayDataView.errmsg = "请填写企业微信用户id,可为多个!";
+                return checkInDayDataView;
+            }
+
+
+            //获取打卡数据 token
+            Access_TokenView access_Token = await GetTokenAsync(2);
+            if (access_Token.errcode != 0)
+            {
+                checkInDayDataView.errcode = access_Token.errcode;
+                checkInDayDataView.errmsg = string.Format("【企业微信】【获取打卡日报数据】【Token】【Msg】{0}", access_Token.errmsg);
+                return checkInDayDataView;
+            }
+
+            string url = string.Format("/cgi-bin/checkin/getcheckin_daydata?access_token={0}&debug=1", access_Token.access_token);
+
+            int days = (int)(endDt - startDt).TotalDays;
+            if (days > 31)
+            {
+                long startTs_One = (startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
+                long endTs_One = (startDt.AddDays(31).ToUniversalTime().Ticks - 621355968000000000) / 10000000;
+                long startTs_Two = (startDt.AddDays(31).ToUniversalTime().Ticks - 621355968000000000) / 10000000;
+                long endTs_Two = (endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
+
+                var checkInData_ReqOne = new
+                {
+                    access_token = access_Token.access_token,
+                    useridlist = useridlist,
+                    starttime = startTs_One,
+                    endtime = endTs_One
+                };
+
+                var jsonOne = System.Text.Json.JsonSerializer.Serialize(checkInData_ReqOne);
+                var contentOne = new StringContent(jsonOne, Encoding.UTF8, "application/json");
+                var create_ReqOne = await _httpClient.PostAsync(url, contentOne);
+                var stringResponseOne = await create_ReqOne.Content.ReadAsStringAsync();
+
+                CheckInDayDataView checkInDayDataViewOne = System.Text.Json.JsonSerializer.Deserialize<CheckInDayDataView>(stringResponseOne,
+                    new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+
+                if (checkInDayDataViewOne.errcode != 0)
+                {
+                    return checkInDayDataViewOne;
+                }
+
+                var checkInData_ReqTwo = new
+                {
+                    access_token = access_Token.access_token,
+                    useridlist = useridlist,
+                    starttime = startTs_Two,
+                    endtime = endTs_Two
+                };
+
+                var jsonTwo = System.Text.Json.JsonSerializer.Serialize(checkInData_ReqTwo);
+                var contentTwo = new StringContent(jsonTwo, Encoding.UTF8, "application/json");
+                var create_ReqTwo = await _httpClient.PostAsync(url, contentTwo);
+                var stringResponseTwo = await create_ReqTwo.Content.ReadAsStringAsync();
+
+                CheckInDayDataView checkInDayDataViewTwo = System.Text.Json.JsonSerializer.Deserialize<CheckInDayDataView>(stringResponseTwo,
+                    new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+                if (checkInDayDataViewTwo.errcode != 0)
+                {
+                    return checkInDayDataViewTwo;
+                }
+                checkInDayDataViewOne.datas.AddRange(checkInDayDataViewTwo.datas);
+                return checkInDayDataViewOne;
+            }
+            else
+            {
+                long startTs = (startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
+                long endTs = (endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
+
+                var checkInData_Req = new
+                {
+                    access_token = access_Token.access_token,
+                    useridlist = useridlist,
+                    starttime = startTs,
+                    endtime = endTs
+                };
+
+                var json = System.Text.Json.JsonSerializer.Serialize(checkInData_Req);
+                var content = new StringContent(json, Encoding.UTF8, "application/json");
+                var create_Req = await _httpClient.PostAsync(url, content);
+                var stringResponse = await create_Req.Content.ReadAsStringAsync();
+
+                checkInDayDataView = System.Text.Json.JsonSerializer.Deserialize<CheckInDayDataView>(stringResponse,
+                    new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+            }
+            
+
+            return checkInDayDataView;
+        }
+
         #endregion
         #endregion
 
 
         #region 审批
         #region 审批
@@ -759,8 +1014,18 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
 
 
             //获取所有打卡补卡,审批 数据 前后范围增加10天
             //获取所有打卡补卡,审批 数据 前后范围增加10天
             DateTime sp_startDt = startDt.AddDays(-10);
             DateTime sp_startDt = startDt.AddDays(-10);
-            DateTime sp_centerDt = sp_startDt.AddDays(30);
+            DateTime sp_centerDt = sp_startDt.AddDays(31);
             DateTime sp_endDt = endDt.AddDays(10);
             DateTime sp_endDt = endDt.AddDays(10);
+            DateTime thisDt = Convert.ToDateTime( DateTime.Now.ToString("yyyy-MM-dd"));
+            if (sp_endDt > thisDt)
+            {
+                sp_endDt = thisDt;
+            }
+            int days = (int)(sp_endDt - sp_startDt).TotalDays;
+            if (days > 31)
+            {
+                sp_centerDt = sp_startDt.AddDays(31);
+            }
 
 
             //获取审批数据 token
             //获取审批数据 token
             Access_TokenView access_Token = await GetTokenAsync(5);
             Access_TokenView access_Token = await GetTokenAsync(5);
@@ -786,8 +1051,8 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
             ApprovalInfo_Request approvalInfoReq = new ApprovalInfo_Request()
             ApprovalInfo_Request approvalInfoReq = new ApprovalInfo_Request()
             {
             {
                 access_token = access_Token.access_token,
                 access_token = access_Token.access_token,
-                starttime = (uint)(startDt - _1970).TotalSeconds,
-                endtime = (uint)(endDt - _1970).TotalSeconds,
+                starttime = (uint)ConvertToTimeSpan(sp_startDt),
+                endtime = (uint)ConvertToTimeSpan(sp_centerDt),
                 filters = filters.ToList(),
                 filters = filters.ToList(),
             };
             };
             var json = System.Text.Json.JsonSerializer.Serialize(approvalInfoReq);
             var json = System.Text.Json.JsonSerializer.Serialize(approvalInfoReq);
@@ -795,36 +1060,63 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
             var create_Req = await _httpClient.PostAsync(url, content);
             var create_Req = await _httpClient.PostAsync(url, content);
             var stringResponse = await create_Req.Content.ReadAsStringAsync();
             var stringResponse = await create_Req.Content.ReadAsStringAsync();
 
 
-            approvalInfoView = System.Text.Json.JsonSerializer.Deserialize<ApprovalInfoView>(stringResponse,
+            ApprovalInfoView approvalInfoView1 = System.Text.Json.JsonSerializer.Deserialize<ApprovalInfoView>(stringResponse,
                 new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
                 new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
 
 
-            List<string> sp_datas = new List<string>();
-            sp_datas.AddRange(approvalInfoView.sp_no_list);
+            if (approvalInfoView1.errcode != 0)
+            {
+                return approvalInfoView1;
+            }
 
 
+            ApprovalInfo_Request approvalInfoReq1 = new ApprovalInfo_Request()
+            {
+                access_token = access_Token.access_token,
+                starttime = (uint)ConvertToTimeSpan(sp_centerDt),
+                endtime = (uint)ConvertToTimeSpan(sp_endDt),
+                filters = filters.ToList(),
+            };
+            var json1 = System.Text.Json.JsonSerializer.Serialize(approvalInfoReq1);
+            var content1 = new StringContent(json1, Encoding.UTF8, "application/json");
+            var create_Req1 = await _httpClient.PostAsync(url, content1);
+            var stringResponse1 = await create_Req1.Content.ReadAsStringAsync();
 
 
-            int index = 0;
-            while (true)
+            ApprovalInfoView approvalInfoView2 = System.Text.Json.JsonSerializer.Deserialize<ApprovalInfoView>(stringResponse1,
+                new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+            if (approvalInfoView2.errcode != 0)
             {
             {
-                if (string.IsNullOrEmpty(approvalInfoView.new_next_cursor)) break;
-                string new_cursor = approvalInfoView.new_next_cursor;
-                approvalInfoReq.new_cursor = new_cursor;
-                var while_json = System.Text.Json.JsonSerializer.Serialize(approvalInfoReq);
-                var while_content = new StringContent(while_json, Encoding.UTF8, "application/json");
-                var while_create_Req = await _httpClient.PostAsync(url, while_content);
-                var while_stringResponse = await while_create_Req.Content.ReadAsStringAsync();
+                return approvalInfoView1;
+            }
 
 
-                approvalInfoView = System.Text.Json.JsonSerializer.Deserialize<ApprovalInfoView>(while_stringResponse,
-                    new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+            approvalInfoView1.sp_no_list.AddRange(approvalInfoView2.sp_no_list);
+
+            //List<string> sp_datas = new List<string>();
+            //sp_datas.AddRange(approvalInfoView1.sp_no_list);
+            //sp_datas.AddRange(approvalInfoView2.sp_no_list);
 
 
-                sp_datas.AddRange(approvalInfoView.sp_no_list); //追加数据
-                index++;
-            }
 
 
-            approvalInfoView.sp_no_list = sp_datas;
+            //int index = 0;
+            //while (true)
+            //{
+            //    if (string.IsNullOrEmpty(approvalInfoView.new_next_cursor)) break;
+            //    string new_cursor = approvalInfoView.new_next_cursor;
+            //    approvalInfoReq.new_cursor = new_cursor;
+            //    var while_json = System.Text.Json.JsonSerializer.Serialize(approvalInfoReq);
+            //    var while_content = new StringContent(while_json, Encoding.UTF8, "application/json");
+            //    var while_create_Req = await _httpClient.PostAsync(url, while_content);
+            //    var while_stringResponse = await while_create_Req.Content.ReadAsStringAsync();
+
+            //    approvalInfoView = System.Text.Json.JsonSerializer.Deserialize<ApprovalInfoView>(while_stringResponse,
+            //        new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
+
+            //    sp_datas.AddRange(approvalInfoView.sp_no_list); //追加数据
+            //    index++;
+            //}
+
+            //approvalInfoView.sp_no_list = sp_datas;
 
 
             sw.Stop();
             sw.Stop();
             approvalInfoView.errmsg = approvalInfoView.errmsg + " 耗时:" + sw.Elapsed.TotalMilliseconds + "ms";
             approvalInfoView.errmsg = approvalInfoView.errmsg + " 耗时:" + sw.Elapsed.TotalMilliseconds + "ms";
-            return approvalInfoView;
+            return approvalInfoView1;
         }
         }
 
 
         /// <summary>
         /// <summary>
@@ -960,5 +1252,10 @@ namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
             return templateDetailView;
             return templateDetailView;
         }
         }
         #endregion
         #endregion
+
+        public long ConvertToTimeSpan(DateTime dt)
+        { 
+            return (dt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
+        }
     }
     }
 }
 }

+ 4 - 0
OASystem/OASystem.Api/appsettings.json

@@ -120,6 +120,10 @@
 
 
   "GrpFileBaseUrl": "http://132.232.92.186:24/",
   "GrpFileBaseUrl": "http://132.232.92.186:24/",
   "GrpFileBasePath": "C:/Server/File/OA2023/Office/GrpFile/",
   "GrpFileBasePath": "C:/Server/File/OA2023/Office/GrpFile/",
+
+  "VisaProgressImageBaseUrl": "http://132.232.92.186:24/",
+  "VisaProgressImageFtpPath": "C:/Server/File/OA2023/Image/Visa/",
+
   //节假日
   //节假日
   "HoliDayDataSource": [
   "HoliDayDataSource": [
     {
     {

+ 3 - 1
OASystem/OASystem.Domain/AutoMappers/_baseMappingProfile.cs

@@ -216,7 +216,9 @@ namespace OASystem.Domain.AutoMappers
             CreateMap<WageSheetInfos, Pm_WageSheet>();
             CreateMap<WageSheetInfos, Pm_WageSheet>();
             CreateMap<SalaryCalculatorSingleDto, Pm_WageSheet>();
             CreateMap<SalaryCalculatorSingleDto, Pm_WageSheet>();
             CreateMap<Pm_WageSheet, WageSheetInfoView>();
             CreateMap<Pm_WageSheet, WageSheetInfoView>();
-            
+
+            CreateMap<WageSheetMonthWorkdaysAddOrEditDto, Pm_WageIssueWorkingDay>();
+            CreateMap<CalendarInfoView, Sys_Calendar>();
             #endregion
             #endregion
         }
         }
     }
     }

+ 17 - 2
OASystem/OASystem.Domain/Dtos/Groups/VisaDto.cs

@@ -6,13 +6,28 @@ using System.Threading.Tasks;
 
 
 namespace OASystem.Domain.Dtos.Groups
 namespace OASystem.Domain.Dtos.Groups
 {
 {
-    public class IOS_VisaDto:DtoBase
+    public class IOS_VisaDto : DtoBase
     {
     {
 
 
     }
     }
 
 
-    public class IOS_VisaCustomerListDto:DtoBase
+    public class IOS_VisaCustomerListDto : DtoBase
     {
     {
         public int diId { get; set; }
         public int diId { get; set; }
     }
     }
+
+    public class IOS_VisaImageListDto : DtoBase {
+        public int visaProgressCustomerId { get; set; }
+        public int picType { get; set; }
+    }
+
+    public class IOS_VisaUploadImageDto : DtoBase
+    {
+        //public string base64DataURL { get; set; }
+        public List<string> base64DataList { get; set; }
+        public string imageName { get; set; }
+        public int picType { get; set; }
+        public int visaProgressCustomerId { get; set; }
+        public int CreateUserId { get; set; }
+    }
 }
 }

+ 8 - 0
OASystem/OASystem.Domain/Dtos/PersonnelModule/WageSheetDto.cs

@@ -14,6 +14,14 @@ namespace OASystem.Domain.Dtos.PersonnelModule
     {
     {
     }
     }
 
 
+    public class WageYearDto
+    {
+        /// <summary>
+        /// 年
+        /// </summary>
+        public int Year { get; set; }
+    }
+
     /// <summary>
     /// <summary>
     /// 工资表单List Dto
     /// 工资表单List Dto
     /// </summary>
     /// </summary>

+ 35 - 0
OASystem/OASystem.Domain/Dtos/PersonnelModule/WageSheetMonthWorkdaysDto.cs

@@ -0,0 +1,35 @@
+using OASystem.Domain.ViewModels.PersonnelModule;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace OASystem.Domain.Dtos.PersonnelModule
+{
+    /// <summary>
+    /// 工资 工作日 Dto
+    /// </summary>
+    public class WageSheetMonthWorkdaysDto
+    {
+    }
+
+    /// <summary>
+    /// 添加 Or 修改
+    /// </summary>
+    public class WageSheetMonthWorkdaysAddOrEditDto : OpBaseDto
+    {
+        /// <summary>
+        /// 年月
+        /// </summary>
+        public string YearMonth { get; set; }
+
+        public string StartDate { get; set; }
+        public string EndDate { get; set; }
+
+        /// <summary>
+        /// 日期信息
+        /// </summary>
+        public List<CalendarInfoView> CalendarInfos { get; set; }
+    }
+}

+ 41 - 0
OASystem/OASystem.Domain/Entities/PersonnelModule/Pm_WageIssueWorkingDay.cs

@@ -0,0 +1,41 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace OASystem.Domain.Entities.PersonnelModule
+{
+    /// <summary>
+    /// 人事 工资
+    /// 月份设置工作日
+    /// </summary>
+    [SugarTable("Pm_WageIssueWorkingDay")]
+    public class Pm_WageIssueWorkingDay: EntityBase
+    {
+        /// <summary>
+        /// 年月
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "varchar(50)")]
+        public string? YearMonth { get; set; }
+
+        /// <summary>
+        /// 工资日期 起
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "varchar(50)")]
+        public string? StartDate { get; set; }
+
+        /// <summary>
+        /// 工资日期 止
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "varchar(50)")]
+        public string? EndDate { get; set; }
+
+        /// <summary>
+        /// 工作日
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "int")]
+        public int Workdays { get; set; }
+
+    }
+}

+ 39 - 0
OASystem/OASystem.Domain/Entities/System/Sys_Calendar.cs

@@ -0,0 +1,39 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace OASystem.Domain.Entities.System
+{
+    /// <summary>
+    /// 日历表
+    /// </summary>
+    [SugarTable("Sys_Calendar")]
+    public class Sys_Calendar : EntityBase
+    {
+        /// <summary>
+        /// 年月日
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "varchar(50)")]
+        public string Dt { get; set; }
+        /// <summary>
+        /// 是否是工作日
+        ///  0 true 1 false
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "bit")]
+        public bool IsWorkDays { get; set; }
+
+        /// <summary>
+        /// 是否是节假日
+        /// 0 true 1 false
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "bit)")]
+        public bool IsHoliDay { get; set; }
+        /// <summary>
+        /// 节假日名称
+        /// </summary>
+        [SugarColumn(IsNullable = true, ColumnDataType = "varchar(50)")]
+        public string HoliDayName { get; set; }
+    }
+}

+ 15 - 9
OASystem/OASystem.Domain/ViewModels/Groups/Grp_DelegationVisaView.cs

@@ -52,15 +52,17 @@ namespace OASystem.Domain.ViewModels.Groups
         {
         {
             get
             get
             {
             {
-                switch (StatusSign) {
-                    case 0: statusstr= "未完成"; break;
-                    case 1: statusstr= "已完成";break;
-                    case 2: statusstr= "已忽略";break;
-                    default: statusstr= "未知";break;
+                switch (StatusSign)
+                {
+                    case 0: statusstr = "未完成"; break;
+                    case 1: statusstr = "已完成"; break;
+                    case 2: statusstr = "已忽略"; break;
+                    default: statusstr = "未知"; break;
                 }
                 }
                 return statusstr;
                 return statusstr;
             }
             }
-            set {
+            set
+            {
                 this.statusstr = value;
                 this.statusstr = value;
             }
             }
         }
         }
@@ -82,8 +84,6 @@ namespace OASystem.Domain.ViewModels.Groups
         }
         }
     }
     }
 
 
-    
-
     public class DelegationVisaVisitCountryView
     public class DelegationVisaVisitCountryView
     {
     {
         public int CountryViewId { get; set; }
         public int CountryViewId { get; set; }
@@ -93,5 +93,11 @@ namespace OASystem.Domain.ViewModels.Groups
         public string PicPath { get; set; }
         public string PicPath { get; set; }
     }
     }
 
 
-
+    public class VisaProgressImageView
+    {
+        public int imageId { get; set; }
+        public string url { get; set; }
+        public string path { get; set; }
+        public string picName { get; set; }
+    }
 }
 }

+ 61 - 0
OASystem/OASystem.Domain/ViewModels/PersonnelModule/WageSheetMonthView.cs

@@ -0,0 +1,61 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace OASystem.Domain.ViewModels.PersonnelModule
+{
+    /// <summary>
+    /// 工资月 List
+    /// </summary>
+    public class WageSheetMonthView
+    {
+        /// <summary>
+        /// 年月
+        /// </summary>
+        public string YearMonth { get; set; }
+
+        /// <summary>
+        /// 开始日期
+        /// </summary>
+        public string StartDate { get; set; }
+
+        /// <summary>
+        /// 结束日期
+        /// </summary>
+        public string EndDate { get; set; }
+
+        /// <summary>
+        /// 当月工作日
+        /// </summary>
+        public int Workdays { get; set; }
+    }
+
+    /// <summary>
+    /// 日期内容Info
+    /// </summary>
+    public class CalendarInfoView
+    {
+        /// <summary>
+        /// 日期 
+        /// Eg:yyyy-MM-dd
+        /// </summary>
+        public string Dt { get; set; }
+
+        /// <summary>
+        /// 是否是工作日
+        /// </summary>
+        public bool IsWorkDay { get; set; }
+
+        /// <summary>
+        /// 是否是节假日
+        /// </summary>
+        public bool IsHoliDay { get; set; }
+
+        /// <summary>
+        /// 节假日名称
+        /// </summary>
+        public string HoliName { get; set; }
+    }
+}

+ 11 - 0
OASystem/OASystem.Domain/ViewModels/PersonnelModule/WageSheetView.cs

@@ -16,6 +16,17 @@ namespace OASystem.Domain.ViewModels.PersonnelModule
 
 
     }
     }
 
 
+    /// <summary>
+    /// 工资年月 View
+    /// </summary>
+    public class WageYearMonthView 
+    {
+        public string YearMonth { get; set; }
+        public string StartDate { get; set; }
+        public string EndDate { get; set; }
+        public int Workdays { get; set; }
+    }
+
     /// <summary>
     /// <summary>
     /// 工资表单详情
     /// 工资表单详情
     /// </summary>
     /// </summary>

+ 14 - 1
OASystem/OASystem.Infrastructure/Repositories/Groups/DelegationVisaRepository.cs

@@ -118,7 +118,7 @@ Group by PicType Order By PicType ", pro.ProgressViewId);
                         case 3: countList.DispatchCount = tempCount; break;
                         case 3: countList.DispatchCount = tempCount; break;
                         case 4: countList.VisaCount = tempCount; break;
                         case 4: countList.VisaCount = tempCount; break;
                         case 5: countList.InvoiceCount = tempCount; break;
                         case 5: countList.InvoiceCount = tempCount; break;
-                        default:break;
+                        default: break;
                     }
                     }
                 }
                 }
                 countList.totalCount();
                 countList.totalCount();
@@ -161,5 +161,18 @@ Group by PicType Order By PicType ", pro.ProgressViewId);
             //return listVisaCustomer;
             //return listVisaCustomer;
         }
         }
 
 
+        public List<VisaProgressImageView> GetVisaProgressImageList(int visaProgressCustomerId, int picType)
+        {
+            string sql1 = string.Format(@" Select 
+Id as imageId,
+PicPath as 'path',
+PicName as picName
+From Grp_VisaProgressCustomerPicture
+Where VisaProgressCustomerId = {0} And PicType = {1} ", visaProgressCustomerId, picType);
+            List<VisaProgressImageView> imageList = _sqlSugar.SqlQueryable<VisaProgressImageView>(sql1).ToList();
+
+            return imageList;
+        }
+
     }
     }
 }
 }

+ 2 - 1
OASystem/OASystem.Infrastructure/Repositories/PersonnelModule/WageSheetRepository.cs

@@ -58,7 +58,8 @@ namespace OASystem.Infrastructure.Repositories.PersonnelModule
                                          Left Join Sys_Users sys_u2 On pm_ws.LastUpdateUserId = sys_u2.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_Company sc On  sys_u1.companyId = sc.Id
                                          Left Join Sys_Department sd On sys_u1.DepId = sd.Id
                                          Left Join Sys_Department sd On sys_u1.DepId = sd.Id
-                                         Where pm_ws.IsDel = 0 And pm_ws.YearMonth = '{0}'", yearMonth);
+                                         Where pm_ws.IsDel = 0 And pm_ws.YearMonth = '{0}'
+                                         Order By UserId Asc ", yearMonth);
             var wageSheetList = await _sqlSugar.SqlQueryable<WageSheetItemInfoView>(sql).ToListAsync();
             var wageSheetList = await _sqlSugar.SqlQueryable<WageSheetItemInfoView>(sql).ToListAsync();
 
 
             if (wageSheetList.Count <= 0 )
             if (wageSheetList.Count <= 0 )