FeeAuditRepository.cs 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873
  1. using EyeSoft.Logging;
  2. using NPOI.SS.Formula.Functions;
  3. using OASystem.Domain;
  4. using OASystem.Domain.Entities.Groups;
  5. using OASystem.Domain.Entities.Resource;
  6. using OASystem.Domain.ViewModels.Groups;
  7. using System;
  8. using System.Collections.Generic;
  9. using System.Linq;
  10. using System.Text;
  11. using System.Threading.Tasks;
  12. namespace OASystem.Infrastructure.Repositories.Groups
  13. {
  14. /// <summary>
  15. /// 费用审核仓储
  16. /// </summary>
  17. public class FeeAuditRepository:BaseRepository<EntityBase,ViewBase>
  18. {
  19. public FeeAuditRepository(SqlSugarClient sqlSugar) : base(sqlSugar)
  20. {
  21. }
  22. /// <summary>
  23. /// 费用自动审核
  24. /// </summary>
  25. /// <param name="feeType">
  26. /// 1.酒店 76
  27. /// 2.op 79
  28. /// 3.其他费用-出行物资 98
  29. /// 4.保险 82
  30. /// </param>
  31. /// <param name="diId">团组Id</param>
  32. /// <param name="dataId">数据Id(模块类型主表Id)</param>
  33. /// <returns></returns>
  34. public async Task<JsonView> FeeAutomaticAudit(int feeType, int diId, int dataId)
  35. {
  36. var _view = new JsonView() { Code = 201, Msg = "自动审核操作失败" };
  37. if (diId < 1) { _view.Msg = MsgTips.DiId; return _view; }
  38. if (dataId < 1) { _view.Msg = MsgTips.Id; return _view; }
  39. var stids = new List<int>() { 17, 66, 91 };
  40. var setData = _sqlSugar.Queryable<Sys_SetData>().Where(x => x.IsDel == 0 && stids.Contains(x.STid)).ToList();
  41. string _teamCurrency = string.Empty;
  42. decimal _teamRate = 0.00M;
  43. var costContents = new List<GroupCostAuditView>();
  44. if (feeType != 4)
  45. {
  46. var groupInfo = _sqlSugar.Queryable<Grp_GroupCostParameter>().Where(x => x.IsDel == 0 && x.DiId == diId).First();
  47. if (groupInfo == null) { _view.Msg = $"团组成本信息未填写!"; return _view; }
  48. _teamCurrency = groupInfo.Currency;
  49. _teamRate = groupInfo.Rate;
  50. //币种验证 统一为currencycode三字码
  51. if (int.TryParse(_teamCurrency, out int currency)) _teamCurrency = setData.Find(x => x.Id == currency)?.Name ?? "";
  52. string costContentSql = $"Select * From Grp_GroupCost";
  53. costContents = _sqlSugar.SqlQueryable<GroupCostAuditView>(costContentSql).Where(x => x.IsDel == 0 && x.Diid == diId).ToList();
  54. if (costContents.Count < 1) { _view.Msg = $"团组成本信息未填写!"; return _view; }
  55. //处理 成本详细信息 日期为空
  56. for (int i = 0; i < costContents.Count; i++)
  57. {
  58. if (string.IsNullOrEmpty(costContents[i].Date))
  59. {
  60. int index = i - 1;
  61. if (index >= 0)
  62. {
  63. costContents[i].Date = costContents[index].Date;
  64. var dtBool = DateTime.TryParse(costContents[i].Date, out DateTime _dateTime);
  65. if (dtBool)
  66. {
  67. costContents[i].CurrTime = _dateTime;
  68. }
  69. }
  70. }
  71. else
  72. {
  73. var dtBool = DateTime.TryParse(costContents[i].Date, out DateTime _dateTime);
  74. if (dtBool)
  75. {
  76. costContents[i].CurrTime = _dateTime;
  77. }
  78. }
  79. }
  80. }
  81. if (feeType == 1)
  82. {
  83. //1089 对冲账或其他 不在审核范围
  84. var hotelCostInfo = _sqlSugar.Queryable<Grp_HotelReservations>().Where(x => x.IsDel == 0 && x.DiId == diId && x.Id == dataId && x.CheckType != 1089).First();
  85. var hotelCostDetails = _sqlSugar.Queryable<Grp_HotelReservationsContent>().Where(x => x.IsDel == 0 && x.DiId == diId && x.HrId == dataId).ToList();
  86. if (hotelCostInfo == null)
  87. {
  88. _view.Msg = $"酒店费用数据未填写";
  89. QuashAudit(76, diId, dataId);
  90. return _view;
  91. }
  92. //酒店费用金额 == 0 不自动审核
  93. if (hotelCostInfo.CardPrice == 0.0000M)
  94. {
  95. _view.Msg = $"酒店费用金额 == 0 不自动审核";
  96. QuashAudit(76, diId, dataId);
  97. return _view;
  98. }
  99. //获取C表汇率
  100. decimal _rate = 1.0000M;
  101. var roomFeeInfo = hotelCostDetails.Where(x => x.PriceType == 1).First();
  102. if (roomFeeInfo == null)
  103. {
  104. _view.Msg = $"酒店房间费用付款数据未填写";
  105. return _view;
  106. }
  107. _rate = roomFeeInfo.Rate == 0.0000M ? 1.0000M : roomFeeInfo.Rate;
  108. bool isAutoAudit = true; //是否自动审核
  109. DateTime checkIn = Convert.ToDateTime(hotelCostInfo.CheckInDate),
  110. checkOut = Convert.ToDateTime(hotelCostInfo.CheckOutDate);
  111. if (checkOut > checkIn) checkOut = checkOut.AddDays(-1); //房费计算,结束日期为前一天
  112. var hotelCostInfos = costContents.Where(x => x.CurrTime >= checkIn && x.CurrTime <= checkOut).ToList();
  113. if (hotelCostInfos.Count < 1) isAutoAudit = false;
  114. decimal otherFee = hotelCostDetails.Where(x => x.PriceType != 1).Sum(x => x.Price * (x.Rate == 0.0000M ? 1.0000M : x.Rate));
  115. if (otherFee > 0) { otherFee /= (checkOut - checkIn).Days; }
  116. var hotelCostInfosGroup = hotelCostInfos.GroupBy(x => x.Date);
  117. foreach (var item in hotelCostInfosGroup)
  118. {
  119. var hotelSingleRoomFee = item.Sum(x => x.HotelSingleRoomFee) * _teamRate; //成本单间费用
  120. var hotelDoubleRoomFee = item.Sum(x => x.HotelDoubleRoomFee) * _teamRate; //成本双人间费用
  121. var hotelSuiteFee = item.Sum(x => x.HotelSuiteFee) * _teamRate; //成本套房费用
  122. var hotelSuiteRoomFee = item.Sum(x => x.HotelSuiteRoomFee) * _teamRate; //成本其他房型间费用
  123. //1.判断费用是否 <= 成本费用
  124. //1.1 判断单间费用
  125. decimal singleRoomPrice = (hotelCostInfo.SingleRoomPrice + otherFee) * _rate; //酒店录入费用
  126. if (singleRoomPrice > 0) if (singleRoomPrice > hotelSingleRoomFee) isAutoAudit = false;
  127. //1.2 判断双人间费用
  128. decimal doubleRoomPrice = (hotelCostInfo.DoubleRoomPrice + otherFee) * _rate;//酒店录入费用
  129. if (doubleRoomPrice > 0) if (doubleRoomPrice > hotelDoubleRoomFee) isAutoAudit = false;
  130. //1.3 判断套房费用
  131. decimal suiteRoomPrice = (hotelCostInfo.SuiteRoomPrice + otherFee) * _rate;//酒店录入费用
  132. if (suiteRoomPrice > 0) if (suiteRoomPrice > hotelSuiteFee) isAutoAudit = false;
  133. //1.4 判断其他房型费用
  134. decimal otherRoomPrice = (hotelCostInfo.OtherRoomPrice + otherFee) * _rate;//酒店录入费用
  135. if (otherRoomPrice > 0) if (otherRoomPrice > hotelSuiteRoomFee) isAutoAudit = false;
  136. }
  137. //2.判断是否自动审核
  138. if (isAutoAudit)
  139. {
  140. var ccpUpdate = _sqlSugar.Updateable<Grp_CreditCardPayment>()
  141. .SetColumns(it => it.IsAuditGM == 3)
  142. .SetColumns(it => it.AuditGMOperate == 4)
  143. .SetColumns(it => it.AuditGMDate == DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"))
  144. .Where(s => s.DIId == diId && s.CTable == 76 && s.CId == dataId)
  145. .ExecuteCommand();
  146. if (ccpUpdate > 0)
  147. {
  148. _view.Code = 200;
  149. _view.Msg = "自动审核执行成功";
  150. return _view;
  151. }
  152. }
  153. else
  154. {
  155. //撤销该条数据的自动审核 --> 该条数据的审核状态是自动审核 3 --> 0
  156. var quashStatus = QuashAudit(76, diId, dataId);
  157. if (quashStatus)
  158. {
  159. _view.Code = 200;
  160. _view.Msg = "费用超团组成本,自动审核撤销成功!";
  161. return _view;
  162. }
  163. }
  164. }
  165. else if (feeType == 2)
  166. {
  167. //1.基础数据参数验证
  168. var priceType = new List<int>() {
  169. 1062 //1062 尾款
  170. };
  171. var opinfos = _sqlSugar.Queryable<Grp_CarTouristGuideGroundReservations>()
  172. .Where(x => x.IsDel == 0 && x.DiId == diId && x.Id == dataId && !priceType.Contains(x.PriceType))
  173. .First();
  174. //var opinfos = _sqlSugar.Queryable<Grp_CarTouristGuideGroundReservations>()
  175. // .Where(x => x.IsDel == 0 && x.DiId == diId && x.Id == dataId )
  176. // .First();
  177. if (opinfos == null)
  178. {
  179. _view.Msg = $"OP费用费用类型属于“尾款”或者 内容未填写";
  180. return _view;
  181. }
  182. //1.含超时费用/超支费用 手动审核
  183. if (opinfos.SelectCheck.Contains("超时") || opinfos.SelectCheck.Contains("超支") || opinfos.SelectCheck.Contains("尾款"))
  184. {
  185. _view.Msg = @$"OP费用含尾款/超支/超时费用,请手动审核";
  186. return _view;
  187. }
  188. //1.参数验证
  189. var opCheckPriceTyeps = opinfos.SelectCheck.Split(',');
  190. var opCheckPriceTyepIds = setData.Where(x => opinfos.SelectCheck.Split(',').Contains(x.Name)).Select(x => x.Id).ToList();
  191. var opContents = _sqlSugar.Queryable<Grp_CarTouristGuideGroundReservationsContent>()
  192. .Where(x => x.IsDel == 0 && x.DiId == diId && x.CTGGRId == dataId && opCheckPriceTyepIds.Contains(x.SId))
  193. .OrderBy(x => x.DatePrice, OrderByType.Asc)
  194. .ToList();
  195. if (opContents.Count < 1)
  196. {
  197. _view.Msg = $"OP费用费用内容未填写";
  198. return _view;
  199. }
  200. //获取C表汇率
  201. decimal _opRate = 1.0000M;
  202. var payInfo = _sqlSugar.Queryable<Grp_CreditCardPayment>().Where(x => x.IsDel == 0 && x.DIId == diId && x.CTable == 79 && x.CId == dataId).First();
  203. if (payInfo == null)
  204. {
  205. _view.Msg = $"OP费用付款数据未填写";
  206. return _view;
  207. }
  208. _opRate = payInfo.DayRate;
  209. string opCurrencyName = setData.Find(x => x.Id == opContents[0].Currency)?.Name ?? "";
  210. //团组、OP币种 验证是否一致
  211. if (opCurrencyName.Equals(_teamCurrency))
  212. {
  213. _opRate = payInfo.DayRate;
  214. _teamRate = payInfo.DayRate;
  215. }
  216. var opBasicDatas = setData.Where(x => x.STid == 17).ToList(); //费用类型基础数据
  217. bool isAutoAudit = true;
  218. if (!DateTime.TryParse(opinfos.ServiceStartTime, out DateTime startDt1) || !DateTime.TryParse(opinfos.ServiceEndTime, out DateTime endDt1))
  219. {
  220. _view.Msg = $"OP费用服务起止日期格式不正确!";
  221. return _view;
  222. }
  223. DateTime startDt = startDt1;
  224. DateTime endDt = endDt1;
  225. var opCostDatas = costContents.Where(it => Convert.ToDateTime(it.Date) >= startDt && Convert.ToDateTime(it.Date) <= endDt).ToList();
  226. if (opCostDatas.Count < 1)
  227. {
  228. _view.Msg = $"该时间段内团组成本未填写!";
  229. return _view;
  230. }
  231. var noAuditFeeTypeIds = new List<int> {
  232. 982 ,//982 车超时费 -- 暂无
  233. 96 ,//96 接送机费 -- 暂无
  234. 97 ,//97 其他费用 -- 暂无
  235. 992 ,//992 住补费用 -- 暂无
  236. 1059,//1059 导游超时费用 -- 暂无
  237. 1070,//1070 尾款金额 -- 暂无
  238. 1071,//1071 其他额外费用 -- 暂无
  239. 1073,//1073 翻译超时费 -- 暂无
  240. 1074,//1074 早餐超支费用 -- 暂无
  241. 1075,//1075 午餐超支费用 -- 暂无
  242. 1076,//1076 晚餐超支费用 -- 暂无
  243. };
  244. //费用类型筛选 包含 feeTypeIds && 包含这些类型费用大于0
  245. var noAuditFeeContents = opContents.Where(x => x.Price > 0 && noAuditFeeTypeIds.Contains(x.SId)).ToList();
  246. if (noAuditFeeContents.Count > 0)
  247. {
  248. _view.Msg = @$"OP费用含尾款/超支/超时费用,请手动审核";
  249. return _view;
  250. }
  251. //2.按天按项 检查费用是否超过预算
  252. var opDayContent = opContents.GroupBy(x => x.DatePrice);
  253. foreach (var item in opDayContent)
  254. {
  255. var opCostInfo = opCostDatas.Where(x => Convert.ToDateTime(x.Date) == item.Key).ToList();
  256. if (opCostInfo.Count < 1) continue;
  257. //车费 91
  258. var opCarCost = item.FirstOrDefault(x => x.SId == 91);
  259. if (opCarCost != null) if (opCarCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.CarFee)) isAutoAudit = false;
  260. //982 车超时费 -- 暂无
  261. //92 导游费
  262. var opGuideCost = item.FirstOrDefault(x => x.SId == 92);
  263. if (opGuideCost != null) if (opGuideCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.GuideFee)) isAutoAudit = false;
  264. //94 导游景点费
  265. var opGuideScenicCost = item.FirstOrDefault(x => x.SId == 94);
  266. if (opGuideScenicCost != null) if (opGuideScenicCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.GuideScenicFee)) isAutoAudit = false;
  267. //95 导游小费
  268. var opGuideTipCost = item.FirstOrDefault(x => x.SId == 95);
  269. if (opGuideTipCost != null) if (opGuideTipCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.GuideTipFee)) isAutoAudit = false;
  270. //983 导游餐补
  271. var opGuideMealCost = item.FirstOrDefault(x => x.SId == 983);
  272. if (opGuideMealCost != null) if (opGuideMealCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.GuideMealFee)) isAutoAudit = false;
  273. //984 导游房补
  274. var opGuideRoomCost = item.FirstOrDefault(x => x.SId == 984);
  275. if (opGuideRoomCost != null) if (opGuideRoomCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.GuideRoomFee)) isAutoAudit = false;
  276. //985 导游交通
  277. var opGuideTrafficCost = item.FirstOrDefault(x => x.SId == 985);
  278. if (opGuideTrafficCost != null) if (opGuideTrafficCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.GuideTrafficFee)) isAutoAudit = false;
  279. //96 接送机费 -- 暂无
  280. //97 其他费用 -- 暂无
  281. //979 司机工资
  282. var opDriverCost = item.FirstOrDefault(x => x.SId == 979);
  283. if (opDriverCost != null) if (opDriverCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.DriverFee)) isAutoAudit = false;
  284. //980 司机小费
  285. var opDriverTipCost = item.FirstOrDefault(x => x.SId == 980);
  286. if (opDriverTipCost != null) if (opDriverTipCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.DriverTipFee)) isAutoAudit = false;
  287. //981 司机餐补
  288. var opDriverMealCost = item.FirstOrDefault(x => x.SId == 981);
  289. if (opDriverMealCost != null) if (opDriverMealCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.DriverMealFee)) isAutoAudit = false;
  290. //988 客户早餐费用
  291. var opClientBreakfastCost = item.FirstOrDefault(x => x.SId == 988);
  292. if (opClientBreakfastCost != null) if (opClientBreakfastCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.ClientBreakfastFee)) isAutoAudit = false;
  293. //93 客户午餐费用
  294. var opClientLunchCost = item.FirstOrDefault(x => x.SId == 93);
  295. if (opClientLunchCost != null) if (opClientLunchCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.ClientLunchFee)) isAutoAudit = false;
  296. //989 客户晚餐费用
  297. var opClientDinnerCost = item.FirstOrDefault(x => x.SId == 989);
  298. if (opClientDinnerCost != null) if (opClientDinnerCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.ClientDinnerFee)) isAutoAudit = false;
  299. //990 景点门票费
  300. var opScenicTicketCost = item.FirstOrDefault(x => x.SId == 990);
  301. if (opScenicTicketCost != null) if (opScenicTicketCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.ScenicTicketFee)) isAutoAudit = false;
  302. //991 饮料/零食/水果
  303. var opDrinkSnackFruitCost = item.FirstOrDefault(x => x.SId == 991);
  304. if (opDrinkSnackFruitCost != null) if (opDrinkSnackFruitCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.DrinkSnackFruitFee)) isAutoAudit = false;
  305. //992 住补费用 -- 暂无
  306. //994 翻译费
  307. var opTranslatorCost = item.FirstOrDefault(x => x.SId == 994);
  308. if (opTranslatorCost != null) if (opTranslatorCost.Price * _opRate > _teamRate * opCostInfo.Sum(x => x.TranslatorFee)) isAutoAudit = false;
  309. //1059 导游超时费用 -- 暂无
  310. //1070 尾款金额 -- 暂无
  311. //1071 其他额外费用 -- 暂无
  312. //1073 翻译超时费 -- 暂无
  313. //1074 早餐超支费用 -- 暂无
  314. //1075 午餐超支费用 -- 暂无
  315. //1076 晚餐超支费用 -- 暂无
  316. }
  317. //更改审核状态
  318. if (isAutoAudit)
  319. {
  320. var ccpUpdate = _sqlSugar.Updateable<Grp_CreditCardPayment>()
  321. .SetColumns(it => it.IsAuditGM == 3)
  322. .SetColumns(it => it.AuditGMOperate == 4)
  323. .SetColumns(it => it.AuditGMDate == DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"))
  324. .Where(s => s.DIId == diId && s.CTable == 79 && s.CId == dataId)
  325. .ExecuteCommand();
  326. if (ccpUpdate > 0)
  327. {
  328. _view.Code = 200;
  329. _view.Msg = "自动审核执行成功!";
  330. return _view;
  331. }
  332. }
  333. else
  334. {
  335. //撤销该条数据的自动审核 --> 该条数据的审核状态是自动审核 3 --> 0
  336. if (QuashAudit(79, diId, dataId))
  337. {
  338. _view.Code = 200;
  339. _view.Msg = "费用超团组成本,自动审核撤销成功!";
  340. return _view;
  341. }
  342. }
  343. }
  344. else if (feeType == 3)
  345. {
  346. #region 出行物资的功能及相关费用自动审核
  347. var isAutoAudit = false;
  348. var currModule = 98; //其他款项
  349. int groupSize = 0; // 团组人数
  350. var groupDetails = await _sqlSugar.Queryable<Grp_DelegationInfo>().Where(x => x.IsDel == 0 && x.Id == diId).FirstAsync();
  351. if (groupDetails != null) groupSize = groupDetails.VisitPNumber;
  352. decimal otherSubTotal = costContents.Sum(x => x.TeFee);
  353. decimal groupCostCNYTotal = costContents.Sum(x => x.TeFee) * _teamRate * groupSize; //团组成本出行物资总金额
  354. if (groupCostCNYTotal <= 0.00M)
  355. {
  356. _view.Msg = $"团组成本出行物资费用数据未填写";
  357. QuashAudit(currModule, diId, dataId);
  358. return _view;
  359. }
  360. var teNames = setData.Where(x => x.STid == 91).Select(x => x.Name).ToList();
  361. var otherFeeDatas = _sqlSugar.Queryable<Grp_DecreasePayments>()
  362. .InnerJoin<Grp_CreditCardPayment>((dp, ccp) => dp.Id == ccp.CId && ccp.CTable == 98 && ccp.IsDel == 0)
  363. .Where((dp, ccp) => dp.IsDel == 0 && dp.DiId == diId)
  364. .ToList();
  365. var ids = new List<int>();
  366. foreach (var item in otherFeeDatas)
  367. {
  368. if (item.PriceName.Contains("、"))
  369. {
  370. var priceNames = item.PriceName.Split('、');
  371. foreach (var priceName in priceNames)
  372. {
  373. if (teNames.Contains(priceName))
  374. {
  375. ids.Add(item.Id);
  376. continue;
  377. }
  378. }
  379. }
  380. else if (teNames.Contains(item.PriceName))
  381. {
  382. ids.Add(item.Id);
  383. }
  384. }
  385. ids = ids.Distinct().ToList();
  386. //验证该费用是不是出行物资费用 是:审核 不是:不审核
  387. if (!ids.Contains(dataId))
  388. {
  389. _view.Msg = $"该费用不属于出行物资,不执行自动审核!";
  390. QuashAudit(currModule, diId, dataId);
  391. return _view;
  392. }
  393. decimal otherFeeCNYTotal = otherFeeDatas.Where(x => ids.Contains(x.Id)).Sum(x => x.FeeTotal); //其他费用出行物资总金额
  394. if (otherFeeCNYTotal <= 0.00M)
  395. {
  396. _view.Msg = $"其他款项出行物资费用数据未填写";
  397. QuashAudit(currModule, diId, dataId);
  398. return _view;
  399. }
  400. if (otherFeeCNYTotal > groupCostCNYTotal)
  401. {
  402. _view.Msg = $"其他款项出行物资费用超出团组成本物资费用";
  403. QuashAudit(currModule, diId, dataId);
  404. return _view;
  405. }
  406. isAutoAudit = true;
  407. //2.判断是否自动审核
  408. if (isAutoAudit)
  409. {
  410. var ccpUpdate = _sqlSugar.Updateable<Grp_CreditCardPayment>()
  411. .SetColumns(it => it.IsAuditGM == 3)
  412. .SetColumns(it => it.AuditGMOperate == 4)
  413. .SetColumns(it => it.AuditGMDate == DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"))
  414. .Where(s => s.DIId == diId && s.CTable == currModule && s.CId == dataId)
  415. .ExecuteCommand();
  416. if (ccpUpdate > 0)
  417. {
  418. _view.Code = 200;
  419. _view.Msg = "自动审核执行成功";
  420. return _view;
  421. }
  422. }
  423. else
  424. {
  425. //撤销该条数据的自动审核 --> 该条数据的审核状态是自动审核 3 --> 0
  426. var quashStatus = QuashAudit(currModule, diId, dataId);
  427. if (quashStatus)
  428. {
  429. _view.Code = 200;
  430. _view.Msg = "费用超团组成本,自动审核撤销成功!";
  431. return _view;
  432. }
  433. }
  434. #endregion
  435. }
  436. else if (feeType == 4)
  437. {
  438. #region 保险费用录入自动审核
  439. var currModule = 82;
  440. var groupDetails = _sqlSugar.Queryable<Grp_DelegationInfo>().Where(x => x.IsDel == 0 && x.Id == diId).First();
  441. if (groupDetails == null)
  442. {
  443. _view.Msg = $"团组信息为空,不可自动审核!";
  444. return _view;
  445. }
  446. var insuranceCountryCostData = _sqlSugar.Queryable<Res_BasicInsuranceCost>().Where(x => x.IsDel == 0).ToList();
  447. if (!insuranceCountryCostData.Any())
  448. {
  449. _view.Msg = $"保险国家费用信息为空,不可自动审核!";
  450. return _view;
  451. }
  452. var insuranceType = _sqlSugar.Queryable<Grp_InsuranceCost>().Where(x => x.IsDel == 0 && x.Id != 2).Select(x => x.Id).ToList();
  453. var insuranceCostData1 = _sqlSugar.Queryable<Grp_Customers>()
  454. .LeftJoin<Grp_CreditCardPayment>((c, ccp) => c.Id == ccp.CId && ccp.CTable == 82)
  455. .Where((c, ccp) => c.IsDel == 0 && c.DiId == diId)
  456. .Select((c, ccp) => new
  457. {
  458. c.Id,
  459. c.Iid,
  460. ccpId = ccp.Id,
  461. CNYPrice = ccp.PayMoney * ccp.DayRate
  462. })
  463. .ToList();
  464. if (!insuranceCostData1.Any())
  465. {
  466. _view.Msg = $"暂无保险数据,不可自动审核!";
  467. return _view;
  468. }
  469. if (insuranceCostData1.Where(x => x.Iid == 2).ToList().Any())
  470. {
  471. _view.Msg = $"保险数据为类型为“新数据请不要选此项”,不可自动审核!";
  472. return _view;
  473. }
  474. #region 处理已审核通过的 “新数据请不要选此项” 数据
  475. var insuranceCostData2 = insuranceCostData1.Where(x => x.Iid == 2).ToList();
  476. if (insuranceCostData2.Any())
  477. {
  478. //_view.Msg = $"保险数据为类型为“新数据请不要选此项”,不可自动审核!";
  479. //return _view;
  480. var ccpIds = insuranceCostData2.Select(x => x.ccpId).ToList();
  481. var auditCcpIds = _sqlSugar.Queryable<Grp_CreditCardPayment>().Where(x => x.IsDel == 0 && ccpIds.Contains(x.Id) && x.IsAuditGM == 3).Select(x => x.Id).ToList();
  482. if (auditCcpIds.Any())
  483. {
  484. var ccpUpd = _sqlSugar.Updateable<Grp_CreditCardPayment>()
  485. .SetColumns(it => it.IsAuditGM == 0)
  486. .SetColumns(it => it.AuditGMOperate == 0)
  487. .SetColumns(it => it.AuditGMDate == string.Empty)
  488. .Where(s => auditCcpIds.Contains(s.Id))
  489. .ExecuteCommand();
  490. }
  491. }
  492. #endregion
  493. //var insuranceCostData = insuranceCostData1.Where(x => insuranceType.Contains(x.Iid)).ToList();
  494. var insuranceCostData = insuranceCostData1.ToList();
  495. var currInsuranceInfo = insuranceCostData.Where(x => x.Id == dataId).FirstOrDefault();
  496. if (!insuranceCostData.Any() && currInsuranceInfo == null)
  497. {
  498. _view.Msg = $"暂无保险数据,不可自动审核!";
  499. return _view;
  500. }
  501. if (currInsuranceInfo.CNYPrice == 0.00M)
  502. {
  503. _view.Msg = $"保险数据未录入费用信息,不可自动审核!";
  504. QuashAudit(currModule, diId, dataId);
  505. return _view;
  506. }
  507. var groupPeopleNum = groupDetails.VisitPNumber;
  508. var visitCountrys = groupDetails.VisitCountry.Split("|").ToList();
  509. if (!visitCountrys.Any())
  510. {
  511. _view.Msg = $"出访国家为空,不可自动审核!";
  512. return _view;
  513. }
  514. var basicCountrys = insuranceCountryCostData.Select(x => x.CountryName).ToList();
  515. var schengenCountry = visitCountrys.Intersect(basicCountrys); //申根国
  516. var unSchengenCountry = visitCountrys.Except(basicCountrys); //非申根国
  517. var schengenCost = insuranceCountryCostData.Where(x => schengenCountry.Contains(x.CountryName)).Sum(x => x.Cost); // 申根国费用
  518. var unSchengenCost = unSchengenCountry.Count() * 35; // 非申根国费用
  519. var groupBudgetCost = (schengenCost + unSchengenCost) * groupPeopleNum;
  520. var groupActialCost = insuranceCostData.Sum(x => x.CNYPrice);
  521. if (groupActialCost > groupBudgetCost)
  522. {
  523. _view.Msg = $"保险费用超出团组成本费用";
  524. QuashAudit(currModule, diId, dataId);
  525. return _view;
  526. }
  527. //自动审核
  528. var ccpUpdate = _sqlSugar.Updateable<Grp_CreditCardPayment>()
  529. .SetColumns(it => it.IsAuditGM == 3)
  530. .SetColumns(it => it.AuditGMOperate == 4)
  531. .SetColumns(it => it.AuditGMDate == DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"))
  532. .Where(s => s.DIId == diId && s.CTable == currModule && s.CId == dataId)
  533. .ExecuteCommand();
  534. if (ccpUpdate > 0)
  535. {
  536. _view.Code = 200;
  537. _view.Msg = "自动审核执行成功";
  538. }
  539. else _view.Msg = "自动审核执行失败";
  540. return _view;
  541. #endregion
  542. }
  543. else _view.Msg = $"请传入有效的feeType参数";
  544. return _view;
  545. }
  546. /// <summary>
  547. /// hotel、op 撤销自动审核的数据
  548. /// </summary>
  549. /// <param name="type">
  550. /// 酒店 76
  551. /// op 79
  552. /// </param>
  553. /// <param name="diId"></param>
  554. /// <param name="dataId"></param>
  555. /// <returns></returns>
  556. private bool QuashAudit(int type, int diId, int dataId)
  557. {
  558. //撤销该条数据的自动审核 --> 该条数据的审核状态是自动审核 3 --> 0
  559. var ccpInfo = _sqlSugar.Queryable<Grp_CreditCardPayment>()
  560. .Where(s => s.DIId == diId && s.CTable == type && s.CId == dataId && s.IsAuditGM == 3)
  561. .First();
  562. if (ccpInfo != null)
  563. {
  564. var ccpUpdate = _sqlSugar.Updateable<Grp_CreditCardPayment>()
  565. .SetColumns(it => it.IsAuditGM == 0)
  566. .SetColumns(it => it.AuditGMOperate == 0)
  567. .SetColumns(it => it.AuditGMDate == string.Empty)
  568. .Where(s => s.Id == ccpInfo.Id)
  569. .ExecuteCommand();
  570. if (ccpUpdate > 0)
  571. {
  572. return true;
  573. }
  574. }
  575. return false;
  576. }
  577. //
  578. /// <summary>
  579. /// 费用自动审核
  580. /// </summary>
  581. /// <param name="feeType">
  582. /// 1.酒店 76
  583. /// 2.op 79
  584. /// </param>
  585. /// <param name="diId">团组Id</param>
  586. /// <param name="dataId">数据Id(模块类型主表Id)</param>
  587. /// <returns></returns>
  588. public string IsOverBudget(int feeType, int diId, int dataId)
  589. {
  590. string _view ="-";
  591. if (diId < 1) { return _view; }
  592. if (dataId < 1) { return _view; }
  593. List<int> stids = new List<int>() { 17, 66 };
  594. var setData = _sqlSugar.Queryable<Sys_SetData>().Where(x => x.IsDel == 0 && stids.Contains(x.STid)).ToList();
  595. string _teamCurrency = string.Empty;
  596. var groupInfo = _sqlSugar.Queryable<Grp_GroupCostParameter>().Where(x => x.IsDel == 0 && x.DiId == diId).First();
  597. if (groupInfo == null) { return _view; }
  598. _teamCurrency = groupInfo.Currency;
  599. //币种验证 统一为currencycode三字码
  600. if (int.TryParse(_teamCurrency, out int currency)) _teamCurrency = setData.Find(x => x.Id == currency)?.Name ?? "";
  601. string costContentSql = $"Select * From Grp_GroupCost";
  602. var costContents = _sqlSugar.SqlQueryable<GroupCostAuditView>(costContentSql).Where(x => x.IsDel == 0 && x.Diid == diId).ToList();
  603. if (costContents.Count < 1) { return _view; }
  604. //处理 成本详细信息 日期为空
  605. for (int i = 0; i < costContents.Count; i++)
  606. {
  607. if (string.IsNullOrEmpty(costContents[i].Date))
  608. {
  609. int index = i - 1;
  610. if (index >= 0)
  611. {
  612. costContents[i].Date = costContents[index].Date;
  613. }
  614. }
  615. }
  616. if (feeType == 1)
  617. {
  618. var hotelCostInfo = _sqlSugar.Queryable<Grp_HotelReservations>().Where(x => x.IsDel == 0 && x.DiId == diId && x.Id == dataId).First();
  619. var hotelCostDetails = _sqlSugar.Queryable<Grp_HotelReservationsContent>().Where(x => x.IsDel == 0 && x.DiId == diId && x.HrId == dataId).ToList();
  620. if (hotelCostInfo == null) return _view;
  621. //获取C表汇率
  622. decimal _rate = 1.0000M;
  623. var roomFeeInfo = hotelCostDetails.Where(x => x.PriceType == 1).First();
  624. if (roomFeeInfo == null) return _view;
  625. _rate = roomFeeInfo.Rate == 0.0000M ? 1.0000M : roomFeeInfo.Rate;
  626. bool isAutoAudit = true; //是否自动审核
  627. DateTime checkIn = Convert.ToDateTime(hotelCostInfo.CheckInDate),
  628. checkOut = Convert.ToDateTime(hotelCostInfo.CheckOutDate);
  629. if (checkOut > checkIn) checkOut = checkOut.AddDays(-1); //房费计算,结束日期为前一天
  630. var hotelCostInfos = costContents.Where(x => Convert.ToDateTime(x.Date) >= checkIn && Convert.ToDateTime(x.Date) <= checkOut).ToList();
  631. if (hotelCostInfos.Count < 1) isAutoAudit = false;
  632. decimal otherFee = hotelCostDetails.Where(x => x.PriceType != 1).Sum(x => x.Price * (x.Rate == 0.0000M ? 1.0000M : x.Rate));
  633. if (otherFee > 0) { otherFee /= (checkOut - checkIn).Days; }
  634. var hotelCostInfosGroup = hotelCostInfos.GroupBy(x => x.Date);
  635. foreach (var item in hotelCostInfosGroup)
  636. {
  637. decimal hotelSingleRoomFee = item.Sum(x => x.HotelSingleRoomFee);
  638. decimal hotelDoubleRoomFee = item.Sum(x => x.HotelDoubleRoomFee);
  639. decimal hotelSuiteFee = item.Sum(x => x.HotelSuiteFee);
  640. decimal hotelSuiteRoomFee = item.Sum(x => x.HotelSuiteRoomFee);
  641. //1.判断费用是否 <= 成本费用
  642. //1.1 判断单间费用
  643. decimal singleRoomPrice = (hotelCostInfo.SingleRoomPrice + otherFee) * _rate;
  644. if (singleRoomPrice > 0) if (singleRoomPrice > hotelSingleRoomFee * _rate) isAutoAudit = false;
  645. //1.2 判断双人间费用
  646. decimal doubleRoomPrice = (hotelCostInfo.DoubleRoomPrice + otherFee) * _rate;
  647. if (doubleRoomPrice > 0) if (doubleRoomPrice > hotelDoubleRoomFee * _rate) isAutoAudit = false;
  648. //1.3 判断套房费用
  649. decimal suiteRoomPrice = (hotelCostInfo.SuiteRoomPrice + otherFee) * _rate;
  650. if (suiteRoomPrice > 0) if (suiteRoomPrice > hotelSuiteFee * _rate) isAutoAudit = false;
  651. //1.4 判断其他房型费用
  652. decimal otherRoomPrice = (hotelCostInfo.OtherRoomPrice + otherFee) * _rate;
  653. if (otherRoomPrice > 0) if (otherRoomPrice > hotelSuiteRoomFee * _rate) isAutoAudit = false;
  654. }
  655. //2.判断是否自动审核
  656. if (isAutoAudit)
  657. {
  658. return $"未超预算";
  659. }
  660. }
  661. else if (feeType == 2)
  662. {
  663. //1.基础数据参数验证
  664. var priceType = new List<int>() { 1062 };
  665. var opinfos = _sqlSugar.Queryable<Grp_CarTouristGuideGroundReservations>()
  666. .Where(x => x.IsDel == 0 && x.DiId == diId && x.Id == dataId && !priceType.Contains(x.PriceType))
  667. .First();
  668. //var opinfos = _sqlSugar.Queryable<Grp_CarTouristGuideGroundReservations>()
  669. // .Where(x => x.IsDel == 0 && x.DiId == diId && x.Id == dataId )
  670. // .First();
  671. if (opinfos == null) return _view;
  672. //1.含超时费用/超支费用 手动审核
  673. if (opinfos.SelectCheck.Contains("超时") || opinfos.SelectCheck.Contains("超支") || opinfos.SelectCheck.Contains("尾款")) return _view;
  674. //1.参数验证
  675. var opCheckPriceTyeps = opinfos.SelectCheck.Split(',');
  676. var opCheckPriceTyepIds = setData.Where(x => opinfos.SelectCheck.Split(',').Contains(x.Name)).Select(x => x.Id).ToList();
  677. var opContents = _sqlSugar.Queryable<Grp_CarTouristGuideGroundReservationsContent>()
  678. .Where(x => x.IsDel == 0 && x.DiId == diId && x.CTGGRId == dataId && opCheckPriceTyepIds.Contains(x.SId))
  679. .OrderBy(x => x.DatePrice, OrderByType.Asc)
  680. .ToList();
  681. if (opContents.Count < 1) return _view;
  682. //获取C表汇率
  683. decimal _rate = 1.0000M;
  684. var payInfo = _sqlSugar.Queryable<Grp_CreditCardPayment>().Where(x => x.IsDel == 0 && x.DIId == diId && x.CTable == 79 && x.Id == dataId).First();
  685. if (payInfo == null) return _view;
  686. _rate = payInfo.DayRate;
  687. string opCurrencyName = setData.Find(x => x.Id == opContents[0].Currency)?.Name ?? "";
  688. var opBasicDatas = setData.Where(x => x.STid == 17).ToList(); //费用类型基础数据
  689. bool isAutoAudit = true;
  690. if (!DateTime.TryParse(opinfos.ServiceStartTime, out DateTime startDt1) || !DateTime.TryParse(opinfos.ServiceEndTime, out DateTime endDt1)) return _view;
  691. DateTime startDt = startDt1;
  692. DateTime endDt = endDt1;
  693. var opCostDatas = costContents.Where(it => Convert.ToDateTime(it.Date) >= startDt && Convert.ToDateTime(it.Date) <= endDt).ToList();
  694. if (opCostDatas.Count < 1) return _view;
  695. var noAuditFeeTypeIds = new List<int> {
  696. 982 ,//982 车超时费 -- 暂无
  697. 96 ,//96 接送机费 -- 暂无
  698. 97 ,//97 其他费用 -- 暂无
  699. 992 ,//992 住补费用 -- 暂无
  700. 1059,//1059 导游超时费用 -- 暂无
  701. 1070,//1070 尾款金额 -- 暂无
  702. 1071,//1071 其他额外费用 -- 暂无
  703. 1073,//1073 翻译超时费 -- 暂无
  704. 1074,//1074 早餐超支费用 -- 暂无
  705. 1075,//1075 午餐超支费用 -- 暂无
  706. 1076,//1076 晚餐超支费用 -- 暂无
  707. };
  708. //费用类型筛选 包含 feeTypeIds && 包含这些类型费用大于0
  709. var noAuditFeeContents = opContents.Where(x => x.Price > 0 && noAuditFeeTypeIds.Contains(x.SId)).ToList();
  710. if (noAuditFeeContents.Count > 0) return _view;
  711. //2.按天按项 检查费用是否超过预算
  712. var opDayContent = opContents.GroupBy(x => x.DatePrice);
  713. foreach (var item in opDayContent)
  714. {
  715. var opCostInfo = opCostDatas.Where(x => Convert.ToDateTime(x.Date) == item.Key).ToList();
  716. if (opCostInfo.Count < 1) continue;
  717. //车费 91
  718. var opCarCost = item.FirstOrDefault(x => x.SId == 91);
  719. if (opCarCost != null) if (opCarCost.Price * _rate > _rate * opCostInfo.Sum(x => x.CarFee)) isAutoAudit = false;
  720. //982 车超时费 -- 暂无
  721. //92 导游费
  722. var opGuideCost = item.FirstOrDefault(x => x.SId == 92);
  723. if (opGuideCost != null) if (opGuideCost.Price * _rate > _rate * opCostInfo.Sum(x => x.GuideFee)) isAutoAudit = false;
  724. //94 导游景点费
  725. var opGuideScenicCost = item.FirstOrDefault(x => x.SId == 94);
  726. if (opGuideScenicCost != null) if (opGuideScenicCost.Price * _rate > _rate * opCostInfo.Sum(x => x.GuideScenicFee)) isAutoAudit = false;
  727. //95 导游小费
  728. var opGuideTipCost = item.FirstOrDefault(x => x.SId == 95);
  729. if (opGuideTipCost != null) if (opGuideTipCost.Price * _rate > _rate * opCostInfo.Sum(x => x.GuideTipFee)) isAutoAudit = false;
  730. //983 导游餐补
  731. var opGuideMealCost = item.FirstOrDefault(x => x.SId == 983);
  732. if (opGuideMealCost != null) if (opGuideMealCost.Price * _rate > _rate * opCostInfo.Sum(x => x.GuideMealFee)) isAutoAudit = false;
  733. //984 导游房补
  734. var opGuideRoomCost = item.FirstOrDefault(x => x.SId == 984);
  735. if (opGuideRoomCost != null) if (opGuideRoomCost.Price * _rate > _rate * opCostInfo.Sum(x => x.GuideRoomFee)) isAutoAudit = false;
  736. //985 导游交通
  737. var opGuideTrafficCost = item.FirstOrDefault(x => x.SId == 985);
  738. if (opGuideTrafficCost != null) if (opGuideTrafficCost.Price * _rate > _rate * opCostInfo.Sum(x => x.GuideTrafficFee)) isAutoAudit = false;
  739. //96 接送机费 -- 暂无
  740. //97 其他费用 -- 暂无
  741. //979 司机工资
  742. var opDriverCost = item.FirstOrDefault(x => x.SId == 979);
  743. if (opDriverCost != null) if (opDriverCost.Price * _rate > _rate * opCostInfo.Sum(x => x.DriverFee)) isAutoAudit = false;
  744. //980 司机小费
  745. var opDriverTipCost = item.FirstOrDefault(x => x.SId == 980);
  746. if (opDriverTipCost != null) if (opDriverTipCost.Price * _rate > _rate * opCostInfo.Sum(x => x.DriverTipFee)) isAutoAudit = false;
  747. //981 司机餐补
  748. var opDriverMealCost = item.FirstOrDefault(x => x.SId == 981);
  749. if (opDriverMealCost != null) if (opDriverMealCost.Price * _rate > _rate * opCostInfo.Sum(x => x.DriverMealFee)) isAutoAudit = false;
  750. //988 客户早餐费用
  751. var opClientBreakfastCost = item.FirstOrDefault(x => x.SId == 988);
  752. if (opClientBreakfastCost != null) if (opClientBreakfastCost.Price * _rate > _rate * opCostInfo.Sum(x => x.ClientBreakfastFee)) isAutoAudit = false;
  753. //93 客户午餐费用
  754. var opClientLunchCost = item.FirstOrDefault(x => x.SId == 93);
  755. if (opClientLunchCost != null) if (opClientLunchCost.Price * _rate > _rate * opCostInfo.Sum(x => x.ClientLunchFee)) isAutoAudit = false;
  756. //989 客户晚餐费用
  757. var opClientDinnerCost = item.FirstOrDefault(x => x.SId == 989);
  758. if (opClientDinnerCost != null) if (opClientDinnerCost.Price * _rate > _rate * opCostInfo.Sum(x => x.ClientDinnerFee)) isAutoAudit = false;
  759. //990 景点门票费
  760. var opScenicTicketCost = item.FirstOrDefault(x => x.SId == 990);
  761. if (opScenicTicketCost != null) if (opScenicTicketCost.Price * _rate > _rate * opCostInfo.Sum(x => x.ScenicTicketFee)) isAutoAudit = false;
  762. //991 饮料/零食/水果
  763. var opDrinkSnackFruitCost = item.FirstOrDefault(x => x.SId == 991);
  764. if (opDrinkSnackFruitCost != null) if (opDrinkSnackFruitCost.Price * _rate > _rate * opCostInfo.Sum(x => x.DrinkSnackFruitFee)) isAutoAudit = false;
  765. //992 住补费用 -- 暂无
  766. //994 翻译费
  767. var opTranslatorCost = item.FirstOrDefault(x => x.SId == 994);
  768. if (opTranslatorCost != null) if (opTranslatorCost.Price * _rate > _rate * opCostInfo.Sum(x => x.TranslatorFee)) isAutoAudit = false;
  769. //1059 导游超时费用 -- 暂无
  770. //1070 尾款金额 -- 暂无
  771. //1071 其他额外费用 -- 暂无
  772. //1073 翻译超时费 -- 暂无
  773. //1074 早餐超支费用 -- 暂无
  774. //1075 午餐超支费用 -- 暂无
  775. //1076 晚餐超支费用 -- 暂无
  776. }
  777. //更改审核状态
  778. if (isAutoAudit) return _view;
  779. }
  780. else return _view;
  781. return _view;
  782. }
  783. }
  784. }