business.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. package proj
  2. import (
  3. "context"
  4. projDao "dashoo.cn/micro/app/dao/proj"
  5. model "dashoo.cn/micro/app/model/proj"
  6. "dashoo.cn/micro/app/service"
  7. "dashoo.cn/opms_libary/myerrors"
  8. "dashoo.cn/opms_libary/utils"
  9. "github.com/gogf/gf/database/gdb"
  10. "github.com/gogf/gf/frame/g"
  11. "github.com/gogf/gf/os/gtime"
  12. "github.com/gogf/gf/util/gconv"
  13. "strconv"
  14. "strings"
  15. )
  16. const (
  17. OpnCreate = "10" // 创建动态
  18. OpnUpdate = "20" // 更新动态
  19. OpnTransfer = "30" // 转移动态
  20. OpnRise = "40" // 升级动态
  21. OpnDrop = "50" // 降级动态
  22. OpnPrimacyContact = "60" // 设置首要联系人动态
  23. OpnStatus = "70" // 更新项目状态动态
  24. OpnAssociation = "80" // 关联联系人动态
  25. OpnDisassociation = "90" // 解除关联联系人动态
  26. )
  27. type businessService struct {
  28. *service.ContextService
  29. Dao *projDao.ProjBusinessDao
  30. }
  31. func NewBusinessService(ctx context.Context) (svc *businessService, err error) {
  32. svc = new(businessService)
  33. if svc.ContextService, err = svc.Init(ctx); err != nil {
  34. return nil, err
  35. }
  36. svc.Dao = projDao.NewProjBusinessDao(svc.Tenant)
  37. return svc, nil
  38. }
  39. func (p *businessService) GetList(req *model.ProjBusinessSearchReq) (total int, businessList []*model.ProjBusiness, err error) {
  40. db := p.Dao.M
  41. if req.NboName != "" {
  42. db = db.WhereLike(p.Dao.Columns.NboName, "%"+req.NboName+"%")
  43. }
  44. if req.CustName != "" {
  45. db = db.WhereLike(p.Dao.Columns.CustName, "%"+req.CustName+"%")
  46. }
  47. if req.SaleName != "" {
  48. db = db.WhereLike(p.Dao.Columns.SaleName, "%"+req.SaleName+"%")
  49. }
  50. if req.NboType != "" {
  51. db = db.Where(p.Dao.Columns.NboType, req.NboType)
  52. }
  53. total, err = db.Count()
  54. if err != nil {
  55. g.Log().Error(err)
  56. err = myerrors.DbError("获取总行数失败。")
  57. return
  58. }
  59. err = db.Page(req.PageNum, req.PageSize).Order("id asc").Scan(&businessList)
  60. return
  61. }
  62. func (p *businessService) GetEntityById(id int64) (business *model.ProjBusiness, err error) {
  63. err = p.Dao.Where(projDao.ProjBusiness.Columns.Id, id).Scan(&business)
  64. return
  65. }
  66. func (p *businessService) GetBusinessProduct(id int64) (productList []*model.ProjBusinessProduct, err error) {
  67. productDao := projDao.NewProjBusinessProductDao(p.Tenant)
  68. err = productDao.Where(productDao.ProjBusinessProductDao.Columns.BusId, id).Scan(&productList)
  69. return
  70. }
  71. func (p *businessService) GetBusinessDynamics(req *model.BusinessReq) (total int, result g.MapStrAny, err error) {
  72. result = make(g.MapStrAny, 0)
  73. dynamicsDao := projDao.NewProjBusinessDynamicsDao(p.Tenant).ProjBusinessDynamicsDao.Where(projDao.ProjBusinessDynamics.Columns.BusId, req.BusId)
  74. total, err = dynamicsDao.Count()
  75. if err != nil {
  76. g.Log().Error(err)
  77. return
  78. }
  79. dynamicsList, err := dynamicsDao.Page(req.GetPage()).Order("created_time desc").All()
  80. if err != nil || dynamicsList == nil {
  81. return
  82. }
  83. // 数据树格式转换
  84. opnDateFlag := gtime.New(dynamicsList[0].OpnDate).Format("Y-m-d")
  85. for k, v := range dynamicsList {
  86. opnDate := gtime.New(v.OpnDate).Format("Y-m-d")
  87. if opnDateFlag == opnDate && k != 0 {
  88. result[opnDate] = append(result[opnDate].(g.ListStrAny), g.Map{
  89. "opnPeople": v.OpnPeople,
  90. "opnDate": v.OpnDate,
  91. "opnType": v.OpnType,
  92. "remark": v.Remark,
  93. "opnContent": gconv.Map(v.OpnContent),
  94. })
  95. } else {
  96. temp := make(g.ListStrAny, 0)
  97. temp = append(temp, g.Map{
  98. "opnPeople": v.OpnPeople,
  99. "opnDate": v.OpnDate,
  100. "opnType": v.OpnType,
  101. "remark": v.Remark,
  102. "opnContent": gconv.Map(v.OpnContent),
  103. })
  104. result[opnDate] = temp
  105. }
  106. }
  107. return
  108. }
  109. func (p *businessService) GetBusinessDynamicsList(req *model.BusinessDynamicsReq) (total int, list []map[string]interface{}, err error) {
  110. dynamicsDao := projDao.NewProjBusinessDynamicsDao(p.Tenant).ProjBusinessDynamicsDao.Where(projDao.ProjBusinessDynamics.Columns.BusId, req.BusId)
  111. if req.OpnType != "" {
  112. dynamicsDao = dynamicsDao.Where(projDao.ProjBusinessDynamics.Columns.OpnType+" = ?", req.OpnType)
  113. }
  114. total, err = dynamicsDao.Count()
  115. if err != nil {
  116. g.Log().Error(err)
  117. return
  118. }
  119. dynamicsList, err := dynamicsDao.Page(req.GetPage()).Order("created_time desc").All()
  120. for _, v := range dynamicsList {
  121. val := gconv.Map(v)
  122. val["opnContent"] = gconv.Map(v.OpnContent)
  123. list = append(list, val)
  124. }
  125. return
  126. }
  127. func (p *businessService) Create(req *model.AddProjBusinessReq) (err error) {
  128. businessData := new(model.ProjBusiness)
  129. if err = gconv.Struct(req, businessData); err != nil {
  130. return
  131. }
  132. contact := g.Map{
  133. projDao.ProjBusinessContact.Columns.ContactId: req.ContactId,
  134. }
  135. service.SetCreatedInfo(contact, p.GetCxtUserId(), p.GetCxtUserName())
  136. products := make([]*model.ProjBusinessProduct, len(req.Products))
  137. if err = gconv.Structs(req.Products, &products); err != nil {
  138. return
  139. }
  140. var totalPrice float64
  141. for _, v := range products {
  142. v.TotalPrice = v.ProdPrice * float64(v.ProdNum)
  143. totalPrice += v.TotalPrice
  144. service.SetCreatedInfo(v, p.GetCxtUserId(), p.GetCxtUserName())
  145. }
  146. businessData.NboCode = "NBO" + strconv.Itoa(int(gtime.Timestamp()))
  147. businessData.NboStatus = "10"
  148. businessData.NboType = "C"
  149. businessData.ApproStatus = "10"
  150. businessData.EstTransPrice = totalPrice
  151. businessData.DeptId = p.GetCxtUserDeptId()
  152. service.SetCreatedInfo(businessData, p.GetCxtUserId(), p.GetCxtUserName())
  153. err = p.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  154. // 添加项目
  155. lastId, err := p.Dao.TX(tx).InsertAndGetId(businessData)
  156. if err != nil {
  157. return err
  158. }
  159. // 创建了联系人
  160. contact[projDao.ProjBusinessContact.Columns.BusId] = lastId
  161. _, err = projDao.NewProjBusinessContactDao(p.Tenant).TX(tx).Insert(contact)
  162. if err != nil {
  163. return err
  164. }
  165. // 处理项目产品信息
  166. for _, v := range products {
  167. v.BusId = int(lastId)
  168. }
  169. // 添加项目产品
  170. _, err = projDao.NewProjBusinessProductDao(p.Tenant).TX(tx).Insert(products)
  171. if err != nil {
  172. return err
  173. }
  174. // 添加项目动态
  175. dynamics := model.ProjBusinessDynamics{
  176. BusId: int(lastId),
  177. OpnType: OpnCreate,
  178. Remark: businessData.Remark,
  179. }
  180. err = p.CreateProjBusinessDynamics(tx, dynamics, businessData)
  181. return err
  182. })
  183. return
  184. }
  185. func (p *businessService) UpdateById(req *model.UpdateProjBusinessReq) error {
  186. record, err := p.Dao.Count("Id", req.Id)
  187. if err != nil {
  188. return err
  189. }
  190. if record == 0 {
  191. return myerrors.TipsError("项目不存在。")
  192. }
  193. businessData := gconv.Map(req)
  194. businessData = utils.MapKeySnakeCamelCase(businessData, "snake")
  195. products := make([]*model.ProjBusinessProduct, len(req.Products))
  196. if err = gconv.Structs(req.Products, &products); err != nil {
  197. return err
  198. }
  199. var totalPrice float64
  200. for _, v := range products {
  201. v.Id = 0
  202. v.BusId = req.Id
  203. v.TotalPrice = v.ProdPrice * float64(v.ProdNum)
  204. totalPrice += v.TotalPrice
  205. service.SetCreatedInfo(v, p.GetCxtUserId(), p.GetCxtUserName())
  206. }
  207. service.SetUpdatedInfo(businessData, p.GetCxtUserId(), p.GetCxtUserName())
  208. err = p.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  209. // 更新项目
  210. _, err = p.Dao.TX(tx).FieldsEx(service.UpdateFieldEx).WherePri(projDao.ProjBusiness.Columns.Id, req.Id).Update(businessData)
  211. if err != nil {
  212. return err
  213. }
  214. // 添加项目产品
  215. _, err = projDao.NewProjBusinessProductDao(p.Tenant).TX(tx).Where(projDao.ProjBusinessProduct.Columns.BusId, req.Id).Delete()
  216. if err != nil {
  217. return err
  218. }
  219. // 添加项目产品
  220. _, err = projDao.NewProjBusinessProductDao(p.Tenant).TX(tx).Insert(products)
  221. if err != nil {
  222. return err
  223. }
  224. // 添加项目动态
  225. dynamics := model.ProjBusinessDynamics{
  226. BusId: req.Id,
  227. OpnType: OpnUpdate,
  228. Remark: req.Remark,
  229. }
  230. err = p.CreateProjBusinessDynamics(tx, dynamics, req)
  231. return err
  232. })
  233. return err
  234. }
  235. func (p *businessService) DeleteByIds(ids []int64) (err error) {
  236. _, err = p.Dao.WhereIn(projDao.ProjBusiness.Columns.Id, ids).Delete()
  237. return
  238. }
  239. // BusinessTransfer 项目转移
  240. func (p *businessService) BusinessTransfer(req *model.BusinessTransferReq) error {
  241. business, err := p.Dao.WherePri(req.Id).One()
  242. if err != nil {
  243. return err
  244. }
  245. if business == nil {
  246. return myerrors.TipsError("项目不存在。")
  247. }
  248. businessMap := g.Map{
  249. p.Dao.Columns.SaleId: req.UserId,
  250. p.Dao.Columns.SaleName: req.UserName,
  251. p.Dao.Columns.Remark: req.Remark,
  252. }
  253. service.SetUpdatedInfo(businessMap, p.GetCxtUserId(), p.GetCxtUserName())
  254. opnContent := businessMap
  255. opnContent["origSaleId"] = business.SaleId
  256. opnContent["origSaleName"] = business.SaleName
  257. err = p.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  258. // 更新项目
  259. _, err = p.Dao.TX(tx).WherePri(projDao.ProjBusiness.Columns.Id, req.Id).Data(businessMap).Update()
  260. if err != nil {
  261. return err
  262. }
  263. // 添加项目动态
  264. dynamics := model.ProjBusinessDynamics{
  265. BusId: req.Id,
  266. OpnType: OpnTransfer,
  267. Remark: req.Remark,
  268. }
  269. err = p.CreateProjBusinessDynamics(tx, dynamics, opnContent)
  270. return err
  271. })
  272. return err
  273. }
  274. // BusinessGradation 项目调级
  275. func (p *businessService) BusinessGradation(req *model.BusinessGradationReq) error {
  276. business, err := p.Dao.Where(projDao.ProjBusiness.Columns.Id, req.Id).One()
  277. if err != nil {
  278. return err
  279. }
  280. if business == nil {
  281. return myerrors.TipsError("项目不存在。")
  282. }
  283. if business.NboType == req.NboType {
  284. return myerrors.TipsError("同级无法进行调级。")
  285. }
  286. opnType := OpnRise
  287. // A < B return -1 项目降级
  288. if strings.Compare(business.NboType, req.NboType) < 0 {
  289. opnType = OpnDrop
  290. }
  291. businessMap := g.Map{
  292. p.Dao.Columns.NboType: req.NboType,
  293. }
  294. service.SetUpdatedInfo(businessMap, p.GetCxtUserId(), p.GetCxtUserName())
  295. err = p.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  296. // 更新项目调级
  297. _, err = p.Dao.TX(tx).WherePri(projDao.ProjBusiness.Columns.Id, req.Id).Data(businessMap).Update()
  298. if err != nil {
  299. return err
  300. }
  301. // 添加项目动态
  302. dynamics := model.ProjBusinessDynamics{
  303. BusId: business.Id,
  304. OpnType: opnType,
  305. Remark: req.Remark,
  306. }
  307. err = p.CreateProjBusinessDynamics(tx, dynamics, g.Map{
  308. "origNboType": business.NboType,
  309. "nboType": req.NboType,
  310. })
  311. return err
  312. })
  313. return err
  314. }
  315. // SetPrimacyContact 项目设置首要联系人
  316. func (p *businessService) SetPrimacyContact(req *model.BusinessPrimacyContactReq) (err error) {
  317. business, err := p.Dao.Where(projDao.ProjBusiness.Columns.Id, req.Id).One()
  318. if err != nil {
  319. return err
  320. }
  321. if business == nil {
  322. return myerrors.TipsError("项目不存在。")
  323. }
  324. businessMap := g.Map{
  325. p.Dao.Columns.ContactId: req.ContactId,
  326. p.Dao.Columns.ContactName: req.ContactName,
  327. p.Dao.Columns.ContactPostion: req.ContactPostion,
  328. p.Dao.Columns.ContactTelephone: req.ContactTelephone,
  329. }
  330. service.SetUpdatedInfo(businessMap, p.GetCxtUserId(), p.GetCxtUserName())
  331. opnContent := businessMap
  332. opnContent["origContactId"] = business.ContactId
  333. opnContent["origContactName"] = business.ContactName
  334. opnContent["origContactPostion"] = business.ContactPostion
  335. opnContent["origContactTelephone"] = business.ContactTelephone
  336. err = p.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  337. // 更新项目
  338. _, err = p.Dao.TX(tx).WherePri(projDao.ProjBusiness.Columns.Id, req.Id).Data(businessMap).Update()
  339. if err != nil {
  340. return err
  341. }
  342. // 添加项目动态
  343. dynamics := model.ProjBusinessDynamics{
  344. BusId: req.Id,
  345. OpnType: OpnPrimacyContact,
  346. Remark: req.Remark,
  347. }
  348. err = p.CreateProjBusinessDynamics(tx, dynamics, opnContent)
  349. return err
  350. })
  351. return err
  352. }
  353. // UpdateBusinessStatus 更新项目状态
  354. func (p *businessService) UpdateBusinessStatus(req *model.UpdateBusinessStatusReq) error {
  355. business, err := p.Dao.WherePri(req.Id).One()
  356. if err != nil {
  357. return err
  358. }
  359. if business == nil {
  360. return myerrors.TipsError("项目不存在。")
  361. }
  362. businessMap := g.Map{
  363. p.Dao.Columns.NboStatus: req.NboStatus,
  364. p.Dao.Columns.Remark: req.Remark,
  365. }
  366. service.SetUpdatedInfo(businessMap, p.GetCxtUserId(), p.GetCxtUserName())
  367. opnContent := businessMap
  368. opnContent["origNboStatus"] = business.NboStatus
  369. err = p.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  370. // 更新项目
  371. _, err = p.Dao.TX(tx).WherePri(projDao.ProjBusiness.Columns.Id, req.Id).Data(businessMap).Update()
  372. if err != nil {
  373. return err
  374. }
  375. // 添加项目动态
  376. dynamics := model.ProjBusinessDynamics{
  377. BusId: req.Id,
  378. OpnType: OpnStatus,
  379. Remark: req.Remark,
  380. }
  381. err = p.CreateProjBusinessDynamics(tx, dynamics, opnContent)
  382. return err
  383. })
  384. return err
  385. }
  386. // CreateProjBusinessDynamics 创建项目动态
  387. func (p *businessService) CreateProjBusinessDynamics(tx *gdb.TX, dynamics model.ProjBusinessDynamics, opnContent interface{}) error {
  388. if v, ok := opnContent.(g.Map); ok {
  389. opnContent = utils.MapKeySnakeCamelCase(v)
  390. }
  391. // 添加项目动态
  392. dynamics.OpnPeopleId = p.GetCxtUserId()
  393. dynamics.OpnPeople = p.GetCxtUserName()
  394. dynamics.OpnDate = gtime.Now()
  395. dynamics.OpnContent = gconv.String(opnContent)
  396. service.SetCreatedInfo(&dynamics, p.GetCxtUserId(), p.GetCxtUserName())
  397. _, err := projDao.NewProjBusinessDynamicsDao(p.Tenant).TX(tx).Insert(&dynamics)
  398. return err
  399. }