pre_payment.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. package contract
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "github.com/gin-gonic/gin"
  6. "github.com/go-playground/validator/v10"
  7. "hongze/fms_api/controller/resp"
  8. "hongze/fms_api/global"
  9. "hongze/fms_api/models/base"
  10. "hongze/fms_api/models/fms"
  11. "hongze/fms_api/models/system"
  12. "hongze/fms_api/services/alarm_msg"
  13. fmsService "hongze/fms_api/services/fms"
  14. "hongze/fms_api/utils"
  15. "strconv"
  16. "time"
  17. )
  18. // // RegisterController 合同登记
  19. type PrePaymentController struct{}
  20. // List
  21. // @Title 到款预登记列表
  22. // @Description 到款预登记列表
  23. // @Param Keyword query string false "关键词"
  24. // @Param StartDate query string false "约定开始时间"
  25. // @Param EndDate query string false "约定结束时间"
  26. // @Param SortType query string true "如何排序,是正序还是倒序,枚举值:`asc 正序`,`desc 倒叙`"
  27. // @Success 200 {object} fms.ContractRegisterItem
  28. // @router /contract/pre_pay/list [get]
  29. func (rg *PrePaymentController) List(c *gin.Context) {
  30. var req fms.PrePayListReq
  31. if e := c.BindQuery(&req); e != nil {
  32. err, ok := e.(validator.ValidationErrors)
  33. if !ok {
  34. resp.FailData("参数解析失败", "Err:"+e.Error(), c)
  35. return
  36. }
  37. resp.FailData("参数解析失败", err.Translate(global.Trans), c)
  38. return
  39. }
  40. cond := `1 = 1`
  41. pars := make([]interface{}, 0)
  42. // 合同编号/客户姓名/销售/实际使用方
  43. if req.Keyword != "" {
  44. kw := "%" + req.Keyword + "%"
  45. cond += ` AND (company_name LIKE ? OR seller_name LIKE ? )`
  46. pars = append(pars, kw, kw)
  47. }
  48. if req.StartDate != "" && req.EndDate != "" {
  49. st := fmt.Sprint(req.StartDate, " 00:00:00")
  50. ed := fmt.Sprint(req.EndDate, " 23:59:59")
  51. cond += ` AND (create_time BETWEEN ? AND ?)`
  52. pars = append(pars, st, ed)
  53. }
  54. page := new(base.Page)
  55. page.SetPageSize(req.PageSize)
  56. page.SetCurrent(req.Current)
  57. sortTypeMap := map[int]bool{0: false, 1: true, 2: false}
  58. page.AddOrderItem(base.OrderItem{Column: "create_time", Asc: sortTypeMap[req.SortType]})
  59. total, list, e := fms.GetPrePayItemPageList(page, cond, pars)
  60. if e != nil {
  61. resp.FailMsg("获取失败", "获取预登记列表失败, Err: "+e.Error(), c)
  62. return
  63. }
  64. registerIds := make([]int, 0)
  65. for i := range list {
  66. registerIds = append(registerIds, list[i].ContractRegisterId)
  67. }
  68. page.SetTotal(total)
  69. baseData := new(base.BaseData)
  70. baseData.SetPage(page)
  71. baseData.SetList(list)
  72. resp.OkData("获取成功", baseData, c)
  73. }
  74. // Add
  75. // @Title 新增到款预登记
  76. // @Description 新增到款预登记
  77. // @Param request body fms.PrepayAddReq true "type json string"
  78. // @Success 200 string "操作成功"
  79. // @router /contract/pre_pay/add [post]
  80. func (rg *PrePaymentController) Add(c *gin.Context) {
  81. req := new(fms.PrepayAddReq)
  82. err := c.ShouldBind(&req)
  83. if err != nil {
  84. errs, ok := err.(validator.ValidationErrors)
  85. if !ok {
  86. resp.FailData("参数解析失败", "Err:"+err.Error(), c)
  87. return
  88. }
  89. resp.FailData("参数解析失败", errs.Translate(global.Trans), c)
  90. return
  91. }
  92. claims, _ := c.Get("adminInfo")
  93. adminInfo := claims.(*system.SysAdmin)
  94. // 日期校验
  95. startDate, e := time.ParseInLocation(utils.FormatDate, req.StartDate, time.Local)
  96. if e != nil {
  97. resp.FailMsg("合同开始日期格式有误", "合同开始日期格式有误, Err: "+e.Error(), c)
  98. return
  99. }
  100. endDate, e := time.ParseInLocation(utils.FormatDate, req.EndDate, time.Local)
  101. if e != nil {
  102. resp.FailMsg("合同结束日期格式有误", "合同结束日期格式有误, Err: "+e.Error(), c)
  103. return
  104. }
  105. // 货币及汇率
  106. rateList, e := fmsService.GetTodayCurrencyRateList()
  107. if e != nil {
  108. resp.FailMsg("操作失败", "获取今日货币汇率失败, Err: "+e.Error(), c)
  109. return
  110. }
  111. var rate float64
  112. for i := range rateList {
  113. if req.CurrencyUnit == rateList[i].Code {
  114. rate = rateList[i].RMBRate
  115. break
  116. }
  117. }
  118. if rate <= 0 {
  119. resp.FailMsg("操作失败", "货币汇率信息有误", c)
  120. return
  121. }
  122. ob := new(fms.ContractPrePayment)
  123. ob.CompanyName = req.CompanyName
  124. ob.SellerId = req.SellerId
  125. ob.SellerName = req.SellerName
  126. ob.CurrencyUnit = req.CurrencyUnit
  127. ob.StartDate = startDate
  128. ob.EndDate = endDate
  129. ob.AdminId = int(adminInfo.AdminId)
  130. ob.AdminName = adminInfo.AdminName
  131. ob.Amount = req.Amount * rate
  132. ob.OriginAmount = req.Amount
  133. ob.CurrencyUnit = req.CurrencyUnit
  134. ob.Remark = req.Remark
  135. ob.Set()
  136. // 新增合同及套餐
  137. if e = ob.Create(); e != nil {
  138. resp.FailMsg("操作失败", "新增合同及套餐失败, Err: "+e.Error(), c)
  139. return
  140. }
  141. resp.Ok("操作成功", c)
  142. }
  143. // Edit
  144. // @Title 编辑到款预登记
  145. // @Description 编辑到款预登记
  146. // @Param request body fms.ContractRegisterEditReq true "type json string"
  147. // @Success 200 string "操作成功"
  148. // @router /contract/pre_pay/edit [post]
  149. func (rg *PrePaymentController) Edit(c *gin.Context) {
  150. req := new(fms.ContractRegisterEditReq)
  151. err := c.ShouldBind(&req)
  152. if err != nil {
  153. errs, ok := err.(validator.ValidationErrors)
  154. if !ok {
  155. resp.FailData("参数解析失败", "Err:"+err.Error(), c)
  156. return
  157. }
  158. resp.FailData("参数解析失败", errs.Translate(global.Trans), c)
  159. return
  160. }
  161. claims, _ := c.Get("adminInfo")
  162. adminInfo := claims.(*system.SysAdmin)
  163. // 日期校验
  164. startDate, e := time.ParseInLocation(utils.FormatDate, req.StartDate, time.Local)
  165. if e != nil {
  166. resp.FailMsg("合同开始日期格式有误", "合同开始日期格式有误, Err: "+e.Error(), c)
  167. return
  168. }
  169. endDate, e := time.ParseInLocation(utils.FormatDate, req.EndDate, time.Local)
  170. if e != nil {
  171. resp.FailMsg("合同结束日期格式有误", "合同结束日期格式有误, Err: "+e.Error(), c)
  172. return
  173. }
  174. signDate, _ := time.ParseInLocation(utils.FormatDate, "0000-00-00", time.Local)
  175. if req.SignDate != "" {
  176. signDateTime, e := time.ParseInLocation(utils.FormatDate, req.SignDate, time.Local)
  177. if e != nil {
  178. resp.FailMsg("合同签订日期格式有误", "合同签订日期格式有误, Err: "+e.Error(), c)
  179. return
  180. }
  181. signDate = signDateTime
  182. }
  183. ob := new(fms.ContractRegister)
  184. item, e := ob.Fetch(req.ContractRegisterId)
  185. if e != nil {
  186. if e == utils.ErrNoRow {
  187. resp.Fail("登记记录不存在或已被删除", c)
  188. return
  189. }
  190. resp.FailMsg("操作失败", "获取合同登记信息失败, Err:"+e.Error(), c)
  191. return
  192. }
  193. // 是否存在相同合同编号的登记
  194. existCond := `contract_code = ?`
  195. existPars := make([]interface{}, 0)
  196. existPars = append(existPars, req.ContractCode)
  197. exist, e := ob.FetchByCondition(existCond, existPars)
  198. if e != nil && e != utils.ErrNoRow {
  199. resp.FailMsg("操作失败", "获取相同登记号失败, Err: "+e.Error(), c)
  200. return
  201. }
  202. if exist != nil && exist.ContractRegisterId > 0 && exist.ContractRegisterId != item.ContractRegisterId {
  203. resp.Fail("合同编号已存在", c)
  204. return
  205. }
  206. originHasPayment := item.HasPayment
  207. updateCols := []string{
  208. "ContractCode", "RelateContractCode", "CrmContractId", "ContractSource", "CompanyName", "ActualCompanyName",
  209. "SellerId", "SellerName", "ContractType", "ContractAmount", "StartDate", "EndDate", "SignDate", "AgreedPayTime",
  210. "ContractStatus", "RegisterStatus", "Remark", "ServiceRemark", "HasPayment", "NewCompany", "ModifyTime",
  211. }
  212. nowTime := time.Now().Local()
  213. item.ContractCode = req.ContractCode
  214. item.RelateContractCode = req.RelateContractCode
  215. item.CrmContractId = req.CrmContractId
  216. item.ContractSource = req.ContractSource
  217. item.CompanyName = req.CompanyName
  218. item.ActualCompanyName = req.ActualCompanyName
  219. item.SellerId = req.SellerId
  220. item.SellerName = req.SellerName
  221. item.ContractType = req.ContractType
  222. item.ContractAmount = req.ContractAmount
  223. item.StartDate = startDate
  224. item.EndDate = endDate
  225. item.SignDate = signDate
  226. item.AgreedPayTime = req.AgreedPayTime
  227. item.ContractStatus = req.ContractStatus
  228. item.RegisterStatus = fms.ContractRegisterStatusIng
  229. item.Remark = req.Remark
  230. item.ServiceRemark = req.ServiceRemark
  231. item.HasPayment = req.HasPayment
  232. item.NewCompany = req.NewCompany
  233. item.ModifyTime = nowTime
  234. // 存在代付的直接完成登记, 且不允许进行开票/到款登记
  235. if req.HasPayment == 1 {
  236. item.RegisterStatus = fms.ContractRegisterStatusComplete
  237. updateCols = append(updateCols, "RegisterStatus")
  238. }
  239. // 开票到款信息
  240. invoiceOB := new(fms.ContractInvoice)
  241. invoiceCond := `contract_register_id = ?`
  242. invoicePars := make([]interface{}, 0)
  243. invoicePars = append(invoicePars, req.ContractRegisterId)
  244. invoiceList, e := invoiceOB.List(invoiceCond, invoicePars, "")
  245. if e != nil {
  246. resp.FailMsg("操作失败", "获取合同开票到款列表失败, Err: "+e.Error(), c)
  247. return
  248. }
  249. invoiceUpdateCols := make([]string, 0)
  250. // 开票到款操作类型: 0-无; 1-更新; 2-删除;
  251. logRemark := req.Remark
  252. invoiceHandleType := 0
  253. if originHasPayment == 0 && req.HasPayment == 1 {
  254. // 若从无代付修改为有代付, 则删除无代付期间新增的所有开票到款登记
  255. invoiceHandleType = 2
  256. } else {
  257. // 修改了货币单位后,同步更新汇率及开票到款的换算金额
  258. if req.CurrencyUnit != item.CurrencyUnit {
  259. rateList, e := fmsService.GetTodayCurrencyRateList()
  260. if e != nil {
  261. resp.FailMsg("操作失败", "获取货币列表及汇率失败, Err: "+e.Error(), c)
  262. return
  263. }
  264. var rate float64
  265. for i := range rateList {
  266. if rateList[i].Code == req.CurrencyUnit {
  267. rate = rateList[i].RMBRate
  268. break
  269. }
  270. }
  271. if rate < 0 {
  272. resp.FailMsg("操作失败", "货币汇率有误", c)
  273. return
  274. }
  275. item.CurrencyUnit = req.CurrencyUnit
  276. item.RMBRate = rate
  277. updateCols = append(updateCols, "CurrencyUnit", "RMBRate")
  278. // 调整开票到款换算后的金额, 保留两位小数
  279. for i := range invoiceList {
  280. invoiceList[i].CurrencyUnit = req.CurrencyUnit
  281. a, _ := strconv.ParseFloat(fmt.Sprintf("%.2f", invoiceList[i].OriginAmount/rate), 64)
  282. invoiceList[i].Amount = a
  283. }
  284. invoiceUpdateCols = append(invoiceUpdateCols, "CurrencyUnit", "Amount")
  285. invoiceHandleType = 1
  286. logRemark = fmt.Sprintf("金额单位由%s修改为%s\n%s", item.CurrencyUnit, req.CurrencyUnit, logRemark)
  287. }
  288. }
  289. // 套餐信息
  290. serviceList, e := fmsService.HandleContractServiceAndDetail(req.ProductId, req.Services, true)
  291. if e != nil {
  292. resp.FailMsg("操作失败", "获取合同套餐详情失败, Err: "+e.Error(), c)
  293. return
  294. }
  295. // 更新合同登记、套餐、开票到款
  296. if e = fms.UpdateContractRegister(item, updateCols, serviceList, invoiceList, invoiceUpdateCols, invoiceHandleType); e != nil {
  297. resp.FailMsg("操作失败", "更新合同及套餐失败, Err: "+e.Error(), c)
  298. return
  299. }
  300. // 校验金额-是否修改状态
  301. go fmsService.CheckContractRegisterAmount(item.ContractRegisterId)
  302. // 操作日志
  303. go func() {
  304. opData := ""
  305. opDataByte, e := json.Marshal(req)
  306. if e != nil {
  307. return
  308. }
  309. opData = string(opDataByte)
  310. logItem := new(fms.ContractRegisterLog)
  311. logItem.ContractRegisterId = item.ContractRegisterId
  312. logItem.AdminId = int(adminInfo.AdminId)
  313. logItem.AdminName = adminInfo.RealName
  314. logItem.OpData = opData
  315. logItem.OpType = fms.ContractRegisterOpTypeEdit
  316. logItem.CreateTime = nowTime
  317. logItem.Remark = logRemark
  318. if e = logItem.Create(); e != nil {
  319. return
  320. }
  321. }()
  322. resp.Ok("操作成功", c)
  323. }
  324. // Del
  325. // @Title 删除到款预登记
  326. // @Description 删除到款预登记
  327. // @Param request body fms.ContractRegisterDelReq true "type json string"
  328. // @Success 200 string "操作成功"
  329. // @router /contract/pre_pay/del [post]
  330. func (rg *PrePaymentController) Del(c *gin.Context) {
  331. req := new(fms.ContractRegisterDelReq)
  332. err := c.ShouldBind(&req)
  333. if err != nil {
  334. errs, ok := err.(validator.ValidationErrors)
  335. if !ok {
  336. resp.FailData("参数解析失败", "Err:"+err.Error(), c)
  337. return
  338. }
  339. resp.FailData("参数解析失败", errs.Translate(global.Trans), c)
  340. return
  341. }
  342. claims, _ := c.Get("adminInfo")
  343. adminInfo := claims.(*system.SysAdmin)
  344. ob := new(fms.ContractRegister)
  345. item, e := ob.Fetch(req.ContractRegisterId)
  346. if e != nil {
  347. if e == utils.ErrNoRow {
  348. resp.Fail("合同登记不存在或已被删除", c)
  349. return
  350. }
  351. resp.FailMsg("获取合同登记失败", "Err:"+e.Error(), c)
  352. return
  353. }
  354. nowTime := time.Now().Local()
  355. item.IsDeleted = 1
  356. item.ModifyTime = nowTime
  357. updateCols := []string{"IsDeleted", "ModifyTime"}
  358. if e = item.Update(updateCols); e != nil {
  359. resp.FailMsg("操作失败", "更新合同登记失败, Err:"+e.Error(), c)
  360. return
  361. }
  362. // 删除对应的开票到款登记
  363. go func() {
  364. if e = fms.DeleteContractInvoicesByRegisterId(item.ContractRegisterId); e != nil {
  365. alarm_msg.SendAlarmMsg(fmt.Sprintf("删除合同登记%d后, 删除开票到款记录失败, ErrMsg: %s", item.ContractRegisterId, e.Error()), 3)
  366. return
  367. }
  368. if e = fms.DeleteInvoicePaymentSummaryByRegisterId(item.ContractRegisterId); e != nil {
  369. alarm_msg.SendAlarmMsg(fmt.Sprintf("删除合同登记%d后, 删除开票到款汇总数据失败, ErrMsg: %s", item.ContractRegisterId, e.Error()), 3)
  370. return
  371. }
  372. }()
  373. // 操作日志
  374. go func() {
  375. opData := ""
  376. opDataByte, e := json.Marshal(req)
  377. if e != nil {
  378. return
  379. }
  380. opData = string(opDataByte)
  381. logItem := new(fms.ContractRegisterLog)
  382. logItem.ContractRegisterId = req.ContractRegisterId
  383. logItem.AdminId = int(adminInfo.AdminId)
  384. logItem.AdminName = adminInfo.RealName
  385. logItem.OpData = opData
  386. logItem.OpType = fms.ContractRegisterOpTypeDel
  387. logItem.CreateTime = nowTime
  388. if e = logItem.Create(); e != nil {
  389. return
  390. }
  391. }()
  392. resp.Ok("操作成功", c)
  393. }