cust_customer.go 41 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457
  1. package cust
  2. import (
  3. "bytes"
  4. "context"
  5. "dashoo.cn/micro/app/model/base"
  6. "dashoo.cn/micro/app/model/contract"
  7. sbase "dashoo.cn/micro/app/service/base"
  8. "dashoo.cn/micro/app/service/partner"
  9. "encoding/json"
  10. "fmt"
  11. "github.com/gogf/gf/container/garray"
  12. "github.com/gogf/gf/util/gvalid"
  13. "math"
  14. "strconv"
  15. "strings"
  16. "time"
  17. "dashoo.cn/opms_libary/myerrors"
  18. "dashoo.cn/opms_libary/plugin/dingtalk/message"
  19. "dashoo.cn/opms_libary/request"
  20. "github.com/360EntSecGroup-Skylar/excelize"
  21. "github.com/gogf/gf/database/gdb"
  22. "github.com/gogf/gf/encoding/gjson"
  23. "github.com/gogf/gf/frame/g"
  24. "github.com/gogf/gf/os/gtime"
  25. "github.com/gogf/gf/util/gconv"
  26. "github.com/mozillazg/go-pinyin"
  27. "dashoo.cn/micro/app/dao/cust"
  28. platdao "dashoo.cn/micro/app/dao/plat"
  29. model "dashoo.cn/micro/app/model/cust"
  30. workflowModel "dashoo.cn/micro/app/model/workflow"
  31. "dashoo.cn/micro/app/service"
  32. workflowService "dashoo.cn/micro/app/service/workflow"
  33. )
  34. type CustomerService struct {
  35. *service.ContextService
  36. Dao *cust.CustCustomerDao
  37. BelongDao *cust.CustCustomerBelongDao
  38. DynamicsDao *cust.CustCustomerDynamicsDao
  39. ContactDao *cust.CustCustomerContactDao
  40. FollowDao *platdao.PlatFollowupDao
  41. BelongServer *CustomerbelongService
  42. ContanctServer *CustomercontactService
  43. }
  44. var isPublic, noPublic = "10", "20" // 公海,非公海
  45. var AllocaTion, OperaTion, Receive, Merge = "10", "20", "30", "40" // 10分配20转移30领取40合并
  46. type OpnType struct {
  47. OperaTion string
  48. }
  49. func NewCustomerService(ctx context.Context) (svc *CustomerService, err error) {
  50. svc = new(CustomerService)
  51. if svc.ContextService, err = svc.Init(ctx); err != nil {
  52. return nil, err
  53. }
  54. svc.Dao = cust.NewCustCustomerDao(svc.Tenant)
  55. svc.BelongDao = cust.NewCustCustomerBelongDao(svc.Tenant)
  56. svc.DynamicsDao = cust.NewCustCustomerDynamicsDao(svc.Tenant)
  57. svc.ContactDao = cust.NewCustCustomerContactDao(svc.Tenant)
  58. svc.FollowDao = platdao.NewPlatFollowupDao(svc.Tenant)
  59. svc.BelongServer, _ = NewCustomerBelongService(ctx)
  60. svc.ContanctServer, _ = NewCustomerContactService(ctx)
  61. return svc, nil
  62. }
  63. // GetList 客户列表列表
  64. func (s *CustomerService) GetList(req *model.CustCustomerSearchReq) (total int, customerList []*model.CustList, err error) {
  65. Model := s.Dao.FieldsEx(s.Dao.C.DeletedTime)
  66. if req.IsPublic {
  67. Model = Model.Where(s.Dao.C.IsPublic, isPublic)
  68. } else {
  69. Model = Model.Where(s.Dao.C.IsPublic, noPublic)
  70. if req.TargetType == "self" {
  71. Model = Model.Where(s.Dao.C.SalesId, s.CxtUser.Id)
  72. } else if req.TargetType == "other" {
  73. Model = Model.Where(s.Dao.C.SalesId, s.DataScope["userIds"]).WhereNot(s.Dao.C.SalesId, s.CxtUser.Id)
  74. } else {
  75. Model = Model.DataScope(s.Ctx, "sales_id").Where(s.Dao.C.IsPublic, noPublic)
  76. }
  77. }
  78. //客户名称
  79. if req.CustName != "" {
  80. Model = Model.Where(s.Dao.C.CustName+" like ?", "%"+req.CustName+"%")
  81. }
  82. //客户编码
  83. if req.CustCode != "" {
  84. Model = Model.Where(s.Dao.C.CustCode+" like ?", "%"+req.CustCode+"%")
  85. }
  86. //客户行业
  87. if req.CustIndustry != "" {
  88. Model = Model.Where(s.Dao.C.CustIndustry, req.CustIndustry)
  89. }
  90. //客户级别
  91. if req.CustLevel != "" {
  92. Model = Model.Where(s.Dao.C.CustLevel, req.CustLevel)
  93. }
  94. //
  95. //if req.FollowUpDate != "" {
  96. // Model = Model.Where(s.Dao.C.FollowUpDate+" like ? ", req.FollowUpDate+"%")
  97. //}
  98. //省份
  99. if req.CustProvince != "" {
  100. Model = Model.Where(s.Dao.C.CustProvince, req.CustProvince)
  101. }
  102. //市
  103. if req.CustCity != "" {
  104. Model = Model.Where(s.Dao.C.CustCity, req.CustCity)
  105. }
  106. //所属销售
  107. if req.SalesName != "" {
  108. Model = Model.Where(s.Dao.C.SalesName+" like ?", "%"+req.SalesName+"%")
  109. }
  110. // 获取日期区间范围内的记录
  111. if req.BeginTime != "" && req.EndTime != "" {
  112. begin := strings.Split(req.BeginTime, " ")[0] + " 00:00:00"
  113. end := strings.Split(req.EndTime, " ")[0] + " 23:59:59"
  114. Model = Model.WhereBetween(s.Dao.C.FollowUpDate, begin, end)
  115. }
  116. if req.IsRemovePage {
  117. Model = Model
  118. } else {
  119. Model = Model.Page(req.GetPage())
  120. }
  121. total, err = Model.Count()
  122. if err != nil {
  123. g.Log().Error(err)
  124. err = myerrors.DbError("获取总行数失败。")
  125. return
  126. }
  127. err = Model.Order("id desc").Scan(&customerList)
  128. if err != nil {
  129. g.Log().Error(err)
  130. return
  131. }
  132. return
  133. }
  134. func (s *CustomerService) customerCode(province, industry string) (string, error) {
  135. sequence, err := service.Sequence(s.Dao.DB, "customer_code")
  136. if err != nil {
  137. return "", err
  138. }
  139. return s.customerProvinceCode(province) + industry + sequence, nil
  140. }
  141. func (s *CustomerService) customerProvinceCode(province string) string {
  142. if province == "重庆市" {
  143. return "CQ"
  144. }
  145. province = strings.Trim(province, "市")
  146. province = strings.Trim(province, "省")
  147. province = strings.Trim(province, "特别行政区")
  148. province = strings.Trim(province, "自治区")
  149. province = strings.Trim(province, "壮族")
  150. province = strings.Trim(province, "回族")
  151. province = strings.Trim(province, "维吾尔")
  152. pinyinArg := pinyin.NewArgs()
  153. pinyinArg.Style = pinyin.FIRST_LETTER
  154. provincePinyin := pinyin.Pinyin(province, pinyinArg)
  155. provinceFirstletter := []string{}
  156. for _, i := range provincePinyin {
  157. provinceFirstletter = append(provinceFirstletter, i[0])
  158. }
  159. provinceCode := strings.Join(provinceFirstletter, "")
  160. provinceCode = strings.ToUpper(provinceCode)
  161. fmt.Println(provincePinyin, provinceFirstletter, provinceCode)
  162. return provinceCode
  163. }
  164. // Create 创建客户
  165. func (s *CustomerService) Create(req *model.CustomerAddSeq) (insertId int64, err error) {
  166. cusTomer := new(model.CustCustomer)
  167. count, err := s.Dao.Where(s.Dao.C.CustName, req.CustName).Count()
  168. if err != nil {
  169. g.Log().Error(err)
  170. return
  171. }
  172. if count > 0 {
  173. return 0, myerrors.TipsError("该客户信息已存在,不可重复添加")
  174. }
  175. if err = gconv.Struct(req, cusTomer); err != nil {
  176. return
  177. }
  178. service.SetCreatedInfo(cusTomer, s.GetCxtUserId(), s.CxtUser.NickName)
  179. custCode, err := s.customerCode(req.CustProvince, req.CustIndustry)
  180. if err != nil {
  181. return 0, err
  182. }
  183. cusTomer.CustCode = custCode
  184. cusTomer.CustStatus = "10"
  185. roles := s.GetCxtUserRoles()
  186. isSales := false
  187. for _, v := range roles {
  188. if v == "SalesEngineer" || v == "ProductLineManager" { // 销售角色
  189. isSales = true
  190. break
  191. }
  192. }
  193. if _, ok := s.DataScope["is_big"]; ok {
  194. isSales = true
  195. }
  196. // 销售角色
  197. if isSales {
  198. cusTomer.IsPublic = noPublic
  199. cusTomer.SalesId = s.GetCxtUserId()
  200. cusTomer.SalesName = s.CxtUser.NickName
  201. cusTomer.CustStatus = "30"
  202. insertId, err = s.Dao.InsertAndGetId(cusTomer)
  203. if err != nil {
  204. g.Log().Error(err)
  205. return 0, err
  206. }
  207. err = s.CreateBelong(gconv.Int(insertId))
  208. return insertId, err
  209. } else {
  210. cusTomer.IsPublic = isPublic
  211. insertId, err = s.Dao.InsertAndGetId(cusTomer)
  212. if err != nil {
  213. g.Log().Error(err)
  214. return 0, err
  215. }
  216. }
  217. return insertId, err
  218. }
  219. func reverseMap(originalMap map[string]string) map[string]string {
  220. reversedMap := make(map[string]string)
  221. for key, value := range originalMap {
  222. reversedMap[value] = key
  223. }
  224. return reversedMap
  225. }
  226. // Create 导入客户
  227. func (s *CustomerService) Import(ctx context.Context, req *contract.ExcelImportReq) (err error) {
  228. validErr := gvalid.CheckStruct(ctx, req, nil)
  229. if validErr != nil {
  230. return myerrors.TipsError(validErr.Current().Error())
  231. }
  232. // 下载文件
  233. buf, err := partner.DownFile(req.ExcelUrl)
  234. if err != nil {
  235. return myerrors.TipsError(fmt.Sprintf("下载 excel 异常 %s", err.Error()))
  236. }
  237. // 解析excel,构造销售目标数据
  238. excelData, err := s.parseExcel(buf)
  239. if err != nil {
  240. return myerrors.TipsError(fmt.Sprintf("解析 excel 异常 %s", err.Error()))
  241. }
  242. //客户类型
  243. idyMap, err := service.GetDictDataByType(ctx, "cust_idy")
  244. idyMap = reverseMap(idyMap)
  245. //客户来源
  246. sourceMap, err := service.GetDictDataByType(ctx, "cust_source")
  247. sourceMap = reverseMap(sourceMap)
  248. //关键字
  249. keywordsMap, err := service.GetDictDataByType(ctx, "customer_bidding_keywords")
  250. keywordsMap = reverseMap(keywordsMap)
  251. svc, err := sbase.NewDistrictService(ctx)
  252. treeList, err := svc.GetProvincesList(0)
  253. fmt.Println("treeList", treeList)
  254. //insertContactList := make([]*model.CustCustomerContactSeq, 0)
  255. contactService, err := NewCustomerContactService(ctx)
  256. if err != nil {
  257. return err
  258. }
  259. CustomerService, err := NewCustomerService(ctx)
  260. if err != nil {
  261. return err
  262. }
  263. yesOrNoMap := map[string]string{
  264. "是": "10",
  265. "否": "20",
  266. }
  267. genderMap := map[string]string{
  268. "男": "10",
  269. "女": "20",
  270. }
  271. e := s.Dao.Transaction(ctx, func(ctx context.Context, tx *gdb.TX) error {
  272. for _, data := range excelData {
  273. var province *base.ProvincesTree
  274. var city *base.ProvincesTree
  275. var region *base.ProvincesTree
  276. keywords := strings.Split(data.Keyword, ",")
  277. for _, keyword := range keywords {
  278. if _, ok := keywordsMap[keyword]; !ok {
  279. return myerrors.TipsError(fmt.Sprintf("关键字 %s 不存在", keyword))
  280. }
  281. }
  282. if data.CustProvince == "" {
  283. return myerrors.TipsError("客户省份不能为空")
  284. }
  285. for _, tree := range treeList {
  286. if tree.DistName == data.CustProvince {
  287. province = tree
  288. if data.CustCity != "" {
  289. for _, cityData := range province.Children {
  290. if cityData.DistName == data.CustCity {
  291. city = cityData
  292. if data.CustRegion != "" {
  293. for _, RegionData := range cityData.Children {
  294. if RegionData.DistName == data.CustRegion {
  295. region = RegionData
  296. }
  297. }
  298. }
  299. }
  300. }
  301. }
  302. }
  303. }
  304. if (province.Id == 0) || (city.Id == 0 && data.CustCity != "") || (region.Id == 0 && data.CustRegion != "") {
  305. return myerrors.TipsError("省市区填写错误,请检查后重试")
  306. }
  307. insertCustomer := &model.CustomerAddSeq{
  308. CustName: data.CustName,
  309. AbbrName: data.AbbrName,
  310. CustAddress: data.CustAddress,
  311. CustProvinceId: province.Id,
  312. CustProvince: data.CustProvince,
  313. CustCityId: city.Id,
  314. CustCity: data.CustCity,
  315. CustRegionId: region.Id,
  316. CustRegion: data.CustRegion,
  317. CustIndustry: idyMap[data.CustIndustry],
  318. CustSource: sourceMap[data.CustSource],
  319. CustDistCode: province.Id,
  320. Remark: data.Remark,
  321. Keyword: keywords,
  322. }
  323. id, err := s.Create(insertCustomer)
  324. if err != nil {
  325. return err
  326. }
  327. s.CreateDynamics("创建客户", req, id)
  328. seq := model.CustCustomerContactSeq{
  329. CustId: int(id),
  330. CuctName: data.CuctName,
  331. CuctGender: genderMap[data.CuctGender],
  332. Telephone: data.Telephone,
  333. Wechat: data.Wechat,
  334. Email: data.Email,
  335. Dept: data.Dept,
  336. Postion: data.Postion,
  337. OfficeLocation: data.OfficeLocation,
  338. IsDecision: yesOrNoMap[data.IsDecision],
  339. Remark: data.ContactRemark,
  340. }
  341. err = contactService.Create(&seq)
  342. if err != nil {
  343. return err
  344. }
  345. CustomerService.CreateDynamics("创建联系人", req, gconv.Int64(seq.CustId))
  346. }
  347. return nil
  348. })
  349. return e
  350. }
  351. // excel解构为数据
  352. func (s CustomerService) parseExcel(b []byte) ([]*model.CustomerAddImport, error) {
  353. f, err := excelize.OpenReader(bytes.NewBuffer(b))
  354. if err != nil {
  355. return nil, err
  356. }
  357. sheet := "Sheet1"
  358. rows := f.GetRows(sheet)
  359. if err != nil {
  360. return nil, err
  361. }
  362. // 客户名称 助计名 所在地区 详细地址 所在省 所在市 所在区县 客户行业 客户级别 客户来源 备注 销售名称 开票抬头 关联客户 姓名 性别 电话 微信 邮箱 部门 职位 办公地点 是否关键决策人 招标关键字
  363. var saleTargets []*model.CustomerAddImport
  364. for _, row := range rows[1:] {
  365. temp := &model.CustomerAddImport{
  366. CustName: row[0],
  367. AbbrName: row[1],
  368. CustAddress: row[2],
  369. CustProvince: row[3],
  370. CustCity: row[4],
  371. CustRegion: row[5],
  372. CustIndustry: row[6],
  373. CustSource: row[7],
  374. Keyword: row[8],
  375. Remark: row[9],
  376. CuctName: row[10],
  377. CuctGender: row[11],
  378. Telephone: row[12],
  379. Wechat: row[13],
  380. Email: row[14],
  381. Dept: row[15],
  382. Postion: row[16],
  383. OfficeLocation: row[17],
  384. IsDecision: row[18],
  385. ContactRemark: row[19],
  386. }
  387. saleTargets = append(saleTargets, temp)
  388. }
  389. return saleTargets, nil
  390. }
  391. // CreateBelong 创建客户归属信息
  392. func (s *CustomerService) CreateBelong(custId int) (err error) {
  393. belong := new(model.CustomerBelongAddSeq)
  394. belong.CustId = custId
  395. belong.SaleName = s.CxtUser.NickName
  396. belong.OpnType = AllocaTion
  397. belong.OpnPeople = s.CxtUser.NickName
  398. err = s.BelongServer.Create(belong)
  399. if err != nil {
  400. g.Log().Error(err)
  401. return
  402. }
  403. return
  404. }
  405. // 删除客户
  406. func (s *CustomerService) DeleteByIds(Ids []int64) (err error) {
  407. customerCount, err := s.Dao.Where(" id in (?)", Ids).Count()
  408. if err != nil {
  409. g.Log().Error(err)
  410. return
  411. }
  412. if customerCount == 0 {
  413. err = myerrors.TipsError("客户信息不存在")
  414. return
  415. }
  416. //删除客户表
  417. s.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  418. //客户表
  419. _, err = s.Dao.TX(tx).Where("id in (?)", Ids).Delete()
  420. if err != nil {
  421. g.Log().Error(err)
  422. return err
  423. }
  424. //客户联系人表
  425. _, err = s.ContactDao.Where("cust_id in (?)", Ids).Delete()
  426. if err != nil {
  427. g.Log().Error(err)
  428. return err
  429. }
  430. //客户归属表
  431. _, err = s.BelongDao.Where("cust_id in (?)", Ids).Delete()
  432. if err != nil {
  433. g.Log().Error(err)
  434. return err
  435. }
  436. return nil
  437. })
  438. return
  439. }
  440. // UpdateById 修改客户
  441. func (s *CustomerService) UpdateById(req *model.UpdateCustomer) (err error) {
  442. //判断数据是否存在
  443. count, err := s.Dao.Where("id = ", req.Id).Count()
  444. if err != nil {
  445. g.Log().Error(err)
  446. return
  447. }
  448. if count == 0 {
  449. return
  450. }
  451. //新的客户名字是否存在
  452. num, err := s.Dao.Where(s.Dao.C.CustName, req.CustName).WhereNot(s.Dao.C.Id, req.Id).Count()
  453. if err != nil {
  454. g.Log().Error(err)
  455. return err
  456. }
  457. if num > 0 {
  458. return myerrors.TipsError(fmt.Sprintf("客户名称[%s]已存在", req.CustName))
  459. }
  460. CustomertData := new(model.CustomerAddSeq)
  461. if err = gconv.Struct(req, CustomertData); err != nil {
  462. return
  463. }
  464. service.SetUpdatedInfo(CustomertData, s.GetCxtUserId(), s.CxtUser.NickName)
  465. _, err = s.Dao.FieldsEx(s.Dao.C.CreatedTime, s.Dao.C.CreatedBy, s.Dao.C.CreatedName, s.Dao.C.Id, s.Dao.C.CustCode, s.Dao.C.SalesName, s.Dao.C.SalesId).
  466. WherePri(s.Dao.C.Id, req.Id).Update(CustomertData)
  467. if err != nil {
  468. g.Log().Error(err)
  469. return
  470. }
  471. return
  472. }
  473. var AssignCustomerRequestProcessCode = "PROC-FE42B2D1-6097-4DE8-8AC5-23541B7D5C8A" // BizCode: 11
  474. var MoveToPubicRequestProcessCode = "PROC-C030BD3D-74A3-4FE5-9EBA-DF3BA9F8AEF9" // BizCode: 12
  475. var TransCustomerRequestProcessCode = "PROC-03BE93D3-9A44-43A1-BA6D-D1BB75702542" // BizCode: 13
  476. // AssignCustomerRequest 领取客户申请
  477. func (s *CustomerService) AssignCustomerRequest(ctx context.Context, req *model.AssignCustomerReq) error {
  478. if req.ApplyRemark == "" {
  479. return myerrors.TipsError("请输入申请说明")
  480. }
  481. data, err := s.Dao.Where("id in (?)", req.Ids).LockShared().All()
  482. if err != nil {
  483. return err
  484. }
  485. if len(data) == 0 {
  486. return myerrors.TipsError("领取用户不能为空")
  487. }
  488. for _, v := range data {
  489. if v.CustStatus != "10" {
  490. return myerrors.TipsError(fmt.Sprintf("客户: %s 已被领取", v.CustName))
  491. }
  492. if v.CustProvince == "" {
  493. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的所在省信息", v.CustName))
  494. }
  495. if v.CustCity == "" {
  496. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的所在市信息", v.CustName))
  497. }
  498. if v.CustIndustry == "" {
  499. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的客户类型信息", v.CustName))
  500. }
  501. }
  502. city := []string{}
  503. cityData := s.DataScope["cust_city_id"]
  504. if cityData != nil {
  505. for _, i := range cityData.([]interface{}) {
  506. city = append(city, i.(string))
  507. }
  508. }
  509. fmt.Println(city)
  510. cust := []*model.CustCustomer{}
  511. noNeedApprovalCust := []*model.CustCustomer{}
  512. noNeedApprovalCustId := []int64{}
  513. for _, v := range data {
  514. if service.StringSlicecontains(city, strconv.Itoa(v.CustCityId)) {
  515. noNeedApprovalCust = append(noNeedApprovalCust, v)
  516. noNeedApprovalCustId = append(noNeedApprovalCustId, int64(v.Id))
  517. } else {
  518. cust = append(cust, v)
  519. }
  520. }
  521. if len(noNeedApprovalCustId) != 0 {
  522. err = s.ChangeCustBelong(noNeedApprovalCustId, req.SalesId, req.SalesName)
  523. if err != nil {
  524. return err
  525. }
  526. req.Receive = AllocaTion
  527. err = s.BatchCreatebelong(data, req)
  528. if err != nil {
  529. return err
  530. }
  531. err = s.CreateDynamics("领取客户", &model.AssignCustomerReq{
  532. Ids: noNeedApprovalCustId,
  533. SalesId: req.SalesId,
  534. SalesName: req.SalesName,
  535. Remark: "",
  536. Receive: Receive,
  537. }, noNeedApprovalCustId...)
  538. if err != nil {
  539. return err
  540. }
  541. }
  542. remark, err := json.Marshal(map[string]string{
  543. "applyUserId": strconv.Itoa(s.GetCxtUserId()),
  544. "applyUserName": s.GetCxtUserName(),
  545. })
  546. if err != nil {
  547. return err
  548. }
  549. workflowSrv, err := workflowService.NewFlowService(ctx)
  550. if err != nil {
  551. return err
  552. }
  553. for _, u := range cust {
  554. bizCode := strconv.Itoa(u.Id) + ":" + strconv.Itoa(s.GetCxtUserId())
  555. applyData := gconv.Map(u)
  556. applyData["abstract"] = fmt.Sprintf("申请人:%v 领取公海客户:%v ", s.GetCxtUserName(), u.CustName)
  557. applyData["applyUser"] = s.GetCxtUserName()
  558. applyData["applyRemark"] = req.ApplyRemark
  559. _, err := workflowSrv.CheckStartProcessInstance(bizCode, workflowModel.CustomerReceive, string(remark), applyData)
  560. if err != nil {
  561. g.Log().Error(err)
  562. return err
  563. }
  564. _, err = s.Dao.Where("id = ?", u.Id).Data(map[string]interface{}{
  565. "is_public": noPublic,
  566. "cust_status": "20",
  567. }).Update()
  568. if err != nil {
  569. return err
  570. }
  571. }
  572. return nil
  573. }
  574. func (s *CustomerService) AssignCustomerRequestApproval(flow *workflowModel.PlatWorkflow, msg *message.MixMessage) error {
  575. remark := map[string]string{}
  576. err := json.Unmarshal([]byte(flow.Remark), &remark)
  577. if err != nil {
  578. return err
  579. }
  580. userName := remark["applyUserName"]
  581. bizCode := strings.Split(flow.BizCode, ":")
  582. if len(bizCode) != 2 {
  583. return fmt.Errorf("客户领取审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  584. }
  585. custId, err := strconv.Atoi(bizCode[0])
  586. if err != nil {
  587. return fmt.Errorf("客户领取审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  588. }
  589. userId, err := strconv.Atoi(bizCode[1])
  590. if err != nil {
  591. return fmt.Errorf("客户领取审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  592. }
  593. cust, err := s.Dao.Where("id = ?", custId).One()
  594. if err != nil {
  595. return err
  596. }
  597. if cust == nil {
  598. return fmt.Errorf("客户不存在:%s Id: %d", flow.BizCode, flow.Id)
  599. }
  600. if msg.ProcessType != "finish" && msg.ProcessType != "terminate" {
  601. return fmt.Errorf("无法识别的 ProcessType :%s", msg.ProcessType)
  602. }
  603. if msg.Result != "agree" && msg.Result != "refuse" && msg.Result != "" {
  604. return fmt.Errorf("无法识别的 Result :%s", msg.Result)
  605. }
  606. if msg.ProcessType == "terminate" {
  607. _, err = s.Dao.Where("id = ?", custId).Data(map[string]interface{}{
  608. "is_public": isPublic,
  609. "cust_status": "10",
  610. }).Update()
  611. return err
  612. }
  613. pass := msg.Result == "agree"
  614. if !pass {
  615. _, err = s.Dao.Where("id = ?", custId).Data(map[string]interface{}{
  616. "is_public": isPublic,
  617. "cust_status": "10",
  618. }).Update()
  619. return err
  620. }
  621. s.CxtUser = &request.UserInfo{Id: userId, UserName: userName}
  622. err = s.ChangeCustBelong([]int64{int64(custId)}, int64(userId), remark["applyUserName"])
  623. if err != nil {
  624. return err
  625. }
  626. err = s.BatchCreatebelong([]*model.CustCustomer{cust}, &model.AssignCustomerReq{
  627. Ids: []int64{int64(custId)},
  628. SalesId: int64(userId),
  629. SalesName: userName,
  630. Remark: "",
  631. Receive: Receive,
  632. })
  633. if err != nil {
  634. return err
  635. }
  636. s.CxtUser = &request.UserInfo{Id: userId, UserName: userName}
  637. return s.CreateDynamics("领取客户", map[string]interface{}{
  638. "userId": userId,
  639. "custId": cust.Id,
  640. }, int64(cust.Id))
  641. }
  642. // 移回公海
  643. func (s *CustomerService) MoveToPublicRequest(ctx context.Context, req *model.MoveToPubicRep) error {
  644. if req.Remark == "" {
  645. return myerrors.TipsError("请输入移回原因")
  646. }
  647. data, err := s.Dao.Where("id in (?)", req.Ids).All()
  648. if err != nil {
  649. return err
  650. }
  651. if len(data) == 0 {
  652. return myerrors.TipsError("移回用户不能为空")
  653. }
  654. cusType, err := service.GetDictDataByType(ctx, "cust_idy")
  655. if err != nil {
  656. return err
  657. }
  658. for _, v := range data {
  659. if v.CustStatus == "10" {
  660. return myerrors.TipsError(fmt.Sprintf("客户: %s 已被移回公海", v.CustName))
  661. }
  662. if v.CustStatus == "20" {
  663. return myerrors.TipsError(fmt.Sprintf("客户: %s 正在等待审批", v.CustName))
  664. }
  665. if v.CustProvince == "" {
  666. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的所在省信息", v.CustName))
  667. }
  668. if v.CustCity == "" {
  669. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的所在市信息", v.CustName))
  670. }
  671. if v.CustIndustry == "" {
  672. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的客户类型信息", v.CustName))
  673. }
  674. }
  675. remark, err := json.Marshal(map[string]string{
  676. "applyUserId": strconv.Itoa(s.GetCxtUserId()),
  677. "applyUserName": s.GetCxtUserName(),
  678. })
  679. if err != nil {
  680. return err
  681. }
  682. workflowSrv, err := workflowService.NewFlowService(ctx)
  683. if err != nil {
  684. return err
  685. }
  686. for _, u := range data {
  687. bizCode := strconv.Itoa(u.Id) + ":" + strconv.Itoa(s.GetCxtUserId())
  688. applyData := gconv.Map(u)
  689. applyData["abstract"] = fmt.Sprintf("申请人:%v 将客户:%v 移回公海", s.GetCxtUserName(), u.CustName)
  690. applyData["custIndustry"] = cusType[u.CustIndustry]
  691. applyData["applyUser"] = s.GetCxtUserName()
  692. applyData["remark"] = req.Remark
  693. _, err := workflowSrv.CheckStartProcessInstance(bizCode, workflowModel.CustomerPublic, string(remark), applyData)
  694. if err != nil {
  695. g.Log().Error(err)
  696. return err
  697. }
  698. _, err = s.Dao.Where("id = ?", u.Id).Data(map[string]interface{}{
  699. "cust_status": "20",
  700. }).Update()
  701. if err != nil {
  702. return err
  703. }
  704. }
  705. return nil
  706. }
  707. // 移入公海回调
  708. func (s *CustomerService) MoveToPublicApproval(flow *workflowModel.PlatWorkflow, msg *message.MixMessage) error {
  709. remark := map[string]string{}
  710. err := json.Unmarshal([]byte(flow.Remark), &remark)
  711. if err != nil {
  712. return err
  713. }
  714. userName := remark["applyUserName"]
  715. bizCode := strings.Split(flow.BizCode, ":")
  716. if len(bizCode) != 2 {
  717. return fmt.Errorf("客户移回审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  718. }
  719. custId, err := strconv.Atoi(bizCode[0])
  720. if err != nil {
  721. return fmt.Errorf("客户移回审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  722. }
  723. userId, err := strconv.Atoi(bizCode[1])
  724. if err != nil {
  725. return fmt.Errorf("客户移回审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  726. }
  727. cust, err := s.Dao.Where("id = ?", custId).One()
  728. if err != nil {
  729. return err
  730. }
  731. if cust == nil {
  732. return fmt.Errorf("客户不存在:%s Id: %d", flow.BizCode, flow.Id)
  733. }
  734. if msg.ProcessType != "finish" && msg.ProcessType != "terminate" {
  735. return fmt.Errorf("无法识别的 ProcessType :%s", msg.ProcessType)
  736. }
  737. if msg.Result != "agree" && msg.Result != "refuse" && msg.Result != "" {
  738. return fmt.Errorf("无法识别的 Result :%s", msg.Result)
  739. }
  740. if msg.ProcessType == "terminate" {
  741. _, err = s.Dao.Where("id = ?", custId).Data(map[string]interface{}{
  742. "cust_status": "30",
  743. }).Update()
  744. return err
  745. }
  746. pass := msg.Result == "agree"
  747. if !pass {
  748. _, err = s.Dao.Where("id = ?", custId).Data(map[string]interface{}{
  749. "cust_status": "30",
  750. }).Update()
  751. return err
  752. }
  753. now := gtime.Now()
  754. err = s.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  755. //更新客户信息
  756. _, err = s.Dao.TX(tx).Data(g.Map{
  757. "cust_status": "10",
  758. "is_public": isPublic,
  759. "sales_id": 0,
  760. "dept_id": 0,
  761. "dept_name": "",
  762. "updated_time": now,
  763. }).Where("id = ?", cust.Id).Update()
  764. if err != nil {
  765. return err
  766. }
  767. //更新销售归属表结束时间
  768. _, err = s.BelongDao.TX(tx).Data(g.Map{
  769. "end_date": now,
  770. }).Where("cust_id = ?", cust.Id).Update()
  771. if err != nil {
  772. return err
  773. }
  774. return nil
  775. })
  776. if err != nil {
  777. return err
  778. }
  779. s.CxtUser = &request.UserInfo{Id: userId, UserName: userName}
  780. return s.CreateDynamics("移入公海", map[string]interface{}{
  781. "userId": userId,
  782. "custId": cust.Id,
  783. }, int64(cust.Id))
  784. }
  785. // SysAdminTransCustomer 系统管理员转移客户(不走审批)
  786. func (s *CustomerService) SysAdminTransCustomer(ctx context.Context, req *model.AssignCustomerReq) error {
  787. // 校验用户是否有修改权限
  788. arr := garray.NewStrArrayFrom(s.CxtUser.Roles, true)
  789. if !arr.Contains("SysAdmin") {
  790. return myerrors.TipsError("权限不足")
  791. }
  792. if req.Remark == "" {
  793. return myerrors.TipsError("请输入转移原因")
  794. }
  795. customerList, err := s.Dao.WherePri(req.Ids).Where(s.Dao.C.IsPublic, "20").All()
  796. if err != nil {
  797. return err
  798. }
  799. if len(customerList) == 0 {
  800. return myerrors.TipsError("转移客户不能为空")
  801. }
  802. var customerIds []int64
  803. for _, item := range customerList {
  804. customerIds = append(customerIds, int64(item.Id))
  805. }
  806. err = s.ChangeCustBelong(customerIds, req.SalesId, req.SalesName)
  807. if err != nil {
  808. return err
  809. }
  810. err = s.BatchCreatebelong(customerList, &model.AssignCustomerReq{
  811. Ids: customerIds,
  812. SalesId: req.SalesId,
  813. SalesName: req.SalesName,
  814. Remark: req.Remark,
  815. Receive: OperaTion,
  816. })
  817. if err != nil {
  818. return err
  819. }
  820. s.CreateDynamics("转移客户", map[string]interface{}{
  821. "toUserId": req.SalesId,
  822. "toUserNickName": req.SalesName,
  823. "remark": "系统管理员转移客户",
  824. }, customerIds...)
  825. return nil
  826. }
  827. // 转移客户
  828. func (s *CustomerService) TransCustomerRequest(ctx context.Context, req *model.AssignCustomerReq) error {
  829. if req.Remark == "" {
  830. return myerrors.TipsError("请输入转移原因")
  831. }
  832. data, err := s.Dao.Where("id in (?)", req.Ids).All()
  833. if err != nil {
  834. return err
  835. }
  836. if len(data) == 0 {
  837. return myerrors.TipsError("转移用户不能为空")
  838. }
  839. for _, v := range data {
  840. if v.CustStatus == "10" {
  841. return myerrors.TipsError(fmt.Sprintf("客户: %s 为公海客户", v.CustName))
  842. }
  843. if v.CustStatus == "20" {
  844. return myerrors.TipsError(fmt.Sprintf("客户: %s 正在等待审批", v.CustName))
  845. }
  846. if v.CustProvince == "" {
  847. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的所在省信息", v.CustName))
  848. }
  849. if v.CustCity == "" {
  850. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的所在市信息", v.CustName))
  851. }
  852. if v.CustIndustry == "" {
  853. return myerrors.TipsError(fmt.Sprintf("请先完善客户: %s 的客户类型信息", v.CustName))
  854. }
  855. }
  856. remark, err := json.Marshal(map[string]string{
  857. "applyUserId": strconv.Itoa(s.GetCxtUserId()),
  858. "applyUserName": s.GetCxtUserName(),
  859. "toUserId": strconv.FormatInt(req.SalesId, 10),
  860. "toUserName": req.SalesName,
  861. })
  862. if err != nil {
  863. return err
  864. }
  865. workflowSrv, err := workflowService.NewFlowService(ctx)
  866. if err != nil {
  867. return err
  868. }
  869. for _, u := range data {
  870. bizCode := strings.Join([]string{
  871. strconv.Itoa(u.Id),
  872. strconv.Itoa(s.GetCxtUserId()),
  873. strconv.FormatInt(req.SalesId, 10),
  874. }, ":")
  875. applyData := gconv.Map(u)
  876. applyData["abstract"] = fmt.Sprintf("申请人:%v 申请转移客户:%v ", s.GetCxtUserName(), u.CustName)
  877. applyData["applyUser"] = s.GetCxtUserName()
  878. applyData["remark"] = req.Remark
  879. _, err := workflowSrv.CheckStartProcessInstance(bizCode, workflowModel.CustomerTrans, string(remark), applyData)
  880. if err != nil {
  881. g.Log().Error(err)
  882. return err
  883. }
  884. _, err = s.Dao.Where("id = ?", u.Id).Data(map[string]interface{}{
  885. "cust_status": "20",
  886. }).Update()
  887. if err != nil {
  888. return err
  889. }
  890. }
  891. return nil
  892. }
  893. // 转移客户回调
  894. func (s *CustomerService) TransCustomerApproval(flow *workflowModel.PlatWorkflow, msg *message.MixMessage) error {
  895. remark := map[string]string{}
  896. err := json.Unmarshal([]byte(flow.Remark), &remark)
  897. if err != nil {
  898. return err
  899. }
  900. userName := remark["applyUserName"]
  901. toUserName := remark["toUserName"]
  902. bizCode := strings.Split(flow.BizCode, ":")
  903. if len(bizCode) != 3 {
  904. return fmt.Errorf("转移客户审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  905. }
  906. custId, err := strconv.Atoi(bizCode[0])
  907. if err != nil {
  908. return fmt.Errorf("转移客户审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  909. }
  910. userId, err := strconv.Atoi(bizCode[1])
  911. if err != nil {
  912. return fmt.Errorf("转移客户审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  913. }
  914. toUserId, err := strconv.Atoi(bizCode[2])
  915. if err != nil {
  916. return fmt.Errorf("转移客户审批 bizCode 不合法:%s Id: %d", flow.BizCode, flow.Id)
  917. }
  918. // user, err := s.UserDao.Where("id = ?", userId).One()
  919. // if err != nil {
  920. // return err
  921. // }
  922. // if user == nil {
  923. // return fmt.Errorf("用户不存在:%s Id: %d", flow.BizCode, flow.Id)
  924. // }
  925. // toUser, err := s.UserDao.Where("id = ?", toUserId).One()
  926. // if err != nil {
  927. // return err
  928. // }
  929. // if toUser == nil {
  930. // return fmt.Errorf("用户不存在:%s Id: %d", flow.BizCode, flow.Id)
  931. // }
  932. cust, err := s.Dao.Where("id = ?", custId).One()
  933. if err != nil {
  934. return err
  935. }
  936. if cust == nil {
  937. return fmt.Errorf("客户不存在:%s Id: %d", flow.BizCode, flow.Id)
  938. }
  939. if msg.ProcessType != "finish" && msg.ProcessType != "terminate" {
  940. return fmt.Errorf("无法识别的 ProcessType :%s", msg.ProcessType)
  941. }
  942. if msg.Result != "agree" && msg.Result != "refuse" && msg.Result != "" {
  943. return fmt.Errorf("无法识别的 Result :%s", msg.Result)
  944. }
  945. if msg.ProcessType == "terminate" {
  946. _, err = s.Dao.Where("id = ?", custId).Data(map[string]interface{}{
  947. "cust_status": "30",
  948. }).Update()
  949. return err
  950. }
  951. pass := msg.Result == "agree"
  952. if !pass {
  953. _, err = s.Dao.Where("id = ?", custId).Data(map[string]interface{}{
  954. "cust_status": "30",
  955. }).Update()
  956. return err
  957. }
  958. s.CxtUser = &request.UserInfo{Id: userId, UserName: userName}
  959. err = s.ChangeCustBelong([]int64{int64(custId)}, int64(toUserId), toUserName)
  960. if err != nil {
  961. return err
  962. }
  963. err = s.BatchCreatebelong([]*model.CustCustomer{cust}, &model.AssignCustomerReq{
  964. Ids: []int64{int64(custId)},
  965. SalesId: int64(toUserId),
  966. SalesName: toUserName,
  967. Remark: "",
  968. Receive: OperaTion,
  969. })
  970. if err != nil {
  971. return err
  972. }
  973. s.CxtUser = &request.UserInfo{Id: userId, UserName: userName}
  974. return s.CreateDynamics("转移客户", map[string]interface{}{
  975. "userId": userId,
  976. "custId": cust.Id,
  977. "toUserId": toUserId,
  978. }, int64(cust.Id))
  979. }
  980. // AssignCustomer 分配客户
  981. func (s *CustomerService) AssignCustomer(ctx context.Context, req *model.AssignCustomerReq) (err error) {
  982. if req.Receive != "" {
  983. return s.AssignCustomerRequest(ctx, req)
  984. }
  985. data, err := s.Dao.Where("id in (?)", req.Ids).LockShared().All()
  986. if err != nil {
  987. g.Log().Error(err)
  988. return
  989. }
  990. if len(data) == 0 {
  991. return myerrors.TipsError("无可分配客户")
  992. }
  993. customerNames := ""
  994. for _, v := range data {
  995. if v.SalesId != 0 {
  996. return myerrors.TipsError(fmt.Sprintf("客户名称[%s]已被领取或分配", v.CustName))
  997. }
  998. if customerNames == "" {
  999. customerNames = v.CustName
  1000. } else {
  1001. customerNames += "、" + v.CustName
  1002. }
  1003. }
  1004. err = s.ChangeCustBelong(req.Ids, req.SalesId, req.SalesName)
  1005. if err != nil {
  1006. return
  1007. }
  1008. req.Receive = AllocaTion
  1009. err = s.BatchCreatebelong(data, req)
  1010. if err != nil {
  1011. return err
  1012. }
  1013. // 消息提醒
  1014. msg := g.MapStrStr{
  1015. "msgTitle": "公海客户分配提醒",
  1016. "msgContent": fmt.Sprintf("客户%v已分配给您,请前往查看", customerNames),
  1017. "msgType": "20",
  1018. "recvUserIds": gconv.String(req.SalesId),
  1019. "msgStatus": "10",
  1020. "sendType": "30",
  1021. }
  1022. if err := service.CreateSystemMessage(msg); err != nil {
  1023. g.Log().Error("公海客户分配提醒异常:", err)
  1024. }
  1025. return s.CreateDynamics("分配客户", req, req.Ids...)
  1026. }
  1027. // GetEntityById 客户详情
  1028. func (s *CustomerService) GetEntityById(Ids []int64) (entityInfo []*model.CustList, err error) {
  1029. Model := s.Dao //
  1030. err = Model.Where(" id in (?)", Ids).Scan(&entityInfo)
  1031. if err != nil {
  1032. g.Log().Error(err)
  1033. return
  1034. }
  1035. return
  1036. }
  1037. // GetCustNameIsExist 判断客户名称是否存在
  1038. func (s *CustomerService) GetCustNameIsExist(req *model.IsExistsCustName) (exist bool, err error) {
  1039. custDao := s.Dao.M
  1040. if req.Id > 0 {
  1041. custDao = custDao.Where("cust_name = ", req.CustName).WhereNot(" id ", req.Id)
  1042. } else {
  1043. custDao = custDao.Where("cust_name = ", req.CustName)
  1044. }
  1045. count, err := custDao.Count()
  1046. if err != nil {
  1047. g.Log().Error(err)
  1048. return
  1049. }
  1050. exist = false
  1051. if count > 0 {
  1052. exist = true
  1053. }
  1054. return
  1055. }
  1056. // CustAbstract 客户摘要
  1057. func (s *CustomerService) CustAbstract(Id int64) (followInfo *model.Follow, err error) {
  1058. count, err := s.FollowDao.Where(s.FollowDao.C.CustId, Id).Count()
  1059. if err != nil {
  1060. g.Log().Error(err)
  1061. return
  1062. }
  1063. followInfo = new(model.Follow)
  1064. followInfo.FollowCount = count
  1065. followTime, err := s.Dao.Fields(s.Dao.C.FollowUpDate, s.Dao.C.CreatedTime).FindOne(Id)
  1066. if err != nil {
  1067. g.Log().Error(err)
  1068. return
  1069. }
  1070. if followTime == nil {
  1071. err = myerrors.TipsError("获取客户信息不存在")
  1072. return
  1073. }
  1074. now := gtime.Now()
  1075. var difference time.Duration
  1076. if followTime.FollowUpDate.IsZero() {
  1077. difference = now.Sub(gtime.New(followTime.CreatedTime))
  1078. } else {
  1079. difference = now.Sub(gtime.New(followTime.FollowUpDate))
  1080. }
  1081. days := difference.Hours() / 24
  1082. if days < 0 {
  1083. followInfo.NotFollowDay = 0
  1084. } else {
  1085. followInfo.NotFollowDay = int(math.Floor(days))
  1086. }
  1087. return
  1088. }
  1089. // // TransCustomer 转移客户
  1090. // func (s *CustomerService) TransCustomer(req *model.AssignCustomerReq) (err error) {
  1091. // data, err := s.Dao.Fields("sales_id,sales_name,id").Where("id in (?)", req.Ids).All()
  1092. // if err != nil {
  1093. // g.Log().Error(err)
  1094. // return err
  1095. // }
  1096. // if len(data) == 0 {
  1097. // return myerrors.TipsError("数据不存在")
  1098. // }
  1099. // err = s.ChangeCustBelong(req.Ids, req.SalesId, req.SalesName)
  1100. // if err != nil {
  1101. // g.Log().Error(err)
  1102. // return
  1103. // }
  1104. // req.Receive = OperaTion
  1105. // s.BatchCreatebelong(data, req)
  1106. // return
  1107. // }
  1108. // ChangeCustBelong 变更客户所属关系
  1109. func (s *CustomerService) ChangeCustBelong(Ids []int64, salesId int64, salesName string) (err error) {
  1110. _, err = s.Dao.Data(g.Map{
  1111. "cust_status": "30",
  1112. "sales_id": salesId,
  1113. "is_public": noPublic,
  1114. "sales_name": salesName,
  1115. "updated_by": s.GetCxtUserId(),
  1116. "updated_name": s.GetCxtUserName(),
  1117. "updated_time": gtime.Now(),
  1118. }).Where("id in (?)", Ids).Update()
  1119. return err
  1120. }
  1121. // CreateDynamics 创建客户动态信息
  1122. func (s *CustomerService) CreateDynamics(opnTpye string, content interface{}, ids ...int64) (err error) {
  1123. datas := make([]*model.CustCustomerDynamics, 0)
  1124. for _, id := range ids {
  1125. dynameics := new(model.CustCustomerDynamics)
  1126. dynameics.CustId = int(id)
  1127. dynameics.OpnPeopleId = s.GetCxtUserId()
  1128. dynameics.OpnPeople = s.GetCxtUserName()
  1129. dynameics.OpnDate = gtime.Now()
  1130. dynameics.OpnType = opnTpye
  1131. v, _ := gjson.Encode(content)
  1132. dynameics.OpnContent = gconv.String(v)
  1133. service.SetCreatedInfo(dynameics, s.GetCxtUserId(), s.GetCxtUserName())
  1134. datas = append(datas, dynameics)
  1135. }
  1136. _, err = s.DynamicsDao.Insert(datas)
  1137. if err != nil {
  1138. g.Log().Error(err)
  1139. return
  1140. }
  1141. return
  1142. }
  1143. // GetDynamicsList 客户动态
  1144. func (s *CustomerService) GetDynamicsList(req *model.CustomerDynameicsReq) (total int, result []interface{}, err error) {
  1145. total, err = s.DynamicsDao.Where("cust_id = ", req.CustId).Count()
  1146. if err != nil {
  1147. g.Log().Error(err)
  1148. return
  1149. }
  1150. dynamics := []*model.CustomerDynameicsRep{}
  1151. err = s.DynamicsDao.Where("cust_id = ", req.CustId).Order("created_time desc").Scan(&dynamics)
  1152. if err != nil {
  1153. g.Log().Error(err)
  1154. return
  1155. }
  1156. dynamicsList := make(map[string][]*model.CustomerDynameicsRep)
  1157. for _, v := range dynamics {
  1158. opnDate := gtime.New(v.OpnDate).Format("Y-m-d")
  1159. dynamicsList[opnDate] = append(dynamicsList[opnDate], &model.CustomerDynameicsRep{
  1160. OpnPeople: v.OpnPeople,
  1161. OpnDate: v.OpnDate,
  1162. OpnType: v.OpnType,
  1163. OpnContent: v.OpnContent,
  1164. })
  1165. }
  1166. result = append(result, dynamicsList)
  1167. return
  1168. }
  1169. // MergeCustomer 合并客户
  1170. func (s *CustomerService) MergeCustomer(req *model.MergeCustomerRep) (err error) {
  1171. //当前目标客户是否存在并取出合并前的销售
  1172. customer, err := s.Dao.Where("id = ", req.Id).One()
  1173. if err != nil {
  1174. g.Log().Error(err)
  1175. return
  1176. }
  1177. if customer == nil {
  1178. return myerrors.TipsError("该客户不存在")
  1179. }
  1180. _, err = s.ContactDao.Where(" cust_id in (?)", req.ChooseId).Delete()
  1181. if err != nil {
  1182. g.Log().Error(err)
  1183. return
  1184. }
  1185. CustomertData := new(model.CustomerAddSeq)
  1186. if err = gconv.Struct(req, CustomertData); err != nil {
  1187. g.Log().Error(err)
  1188. return
  1189. }
  1190. service.SetUpdatedInfo(CustomertData, s.GetCxtUserId(), s.GetCxtUserName())
  1191. _, err = s.Dao.FieldsEx(s.Dao.C.CreatedTime, s.Dao.C.CreatedBy,
  1192. s.Dao.C.CreatedName, s.Dao.C.Id,
  1193. s.Dao.C.CustCode).WherePri(s.Dao.C.Id, req.Id).Update(CustomertData)
  1194. if err != nil {
  1195. g.Log().Error(err)
  1196. return
  1197. }
  1198. //删除被合并的客户信息
  1199. _, err = s.Dao.Where(" id in (?)", req.ChooseId).Delete()
  1200. if err != nil {
  1201. g.Log().Error(err)
  1202. return
  1203. }
  1204. //删除 所选客户销售联系人
  1205. _, err = s.BelongDao.Where(" cust_id in (?)", req.ChooseId).Delete()
  1206. if err != nil {
  1207. g.Log().Error(err)
  1208. return
  1209. }
  1210. //插入一条合并成功的归属记录
  1211. //更新销售归属表销售结束时间
  1212. _, err = s.BelongDao.Data(
  1213. g.Map{
  1214. "updated_by": s.GetCxtUserId(),
  1215. "updated_name": s.GetCxtUserName(),
  1216. "end_date": gtime.Now(),
  1217. }).WhereIn(s.BelongDao.C.CustId, req.Id).Update()
  1218. if err != nil {
  1219. g.Log().Error(err)
  1220. return
  1221. }
  1222. req.CustomerBelongAddSeq.CustId = int(req.Id)
  1223. req.CustomerBelongAddSeq.OpnType = Merge
  1224. req.CustomerBelongAddSeq.OpnPeople = s.GetCxtUserName()
  1225. req.CustomerBelongAddSeq.OrigSaleName = customer.SalesName
  1226. req.CustomerBelongAddSeq.SaleName = req.SalesName
  1227. s.BelongServer.Create(req.CustomerBelongAddSeq)
  1228. return
  1229. }
  1230. // CreateContact 联系人(合并)预留
  1231. func (s *CustomerService) CreateContact(Ids []int64, req *model.CustCustomerContactSeq) (err error) {
  1232. _, err = s.ContactDao.Where(" cust_id in (?)", Ids).Delete()
  1233. if err != nil {
  1234. g.Log().Error(err)
  1235. return
  1236. }
  1237. s.ContanctServer.Create(req)
  1238. return
  1239. }
  1240. // BatchCreatebelong 批量插入客户归属记录表
  1241. func (s *CustomerService) BatchCreatebelong(rep []*model.CustCustomer, req *model.AssignCustomerReq, n ...interface{}) (err error) {
  1242. //更新销售归属表销售结束时间
  1243. _, err = s.BelongDao.Data(
  1244. g.Map{
  1245. "updated_by": s.GetCxtUserId(),
  1246. "updated_name": s.GetCxtUserName(),
  1247. "end_date": gtime.Now(),
  1248. }).WhereIn(s.BelongDao.C.CustId, req.Ids).Update()
  1249. if err != nil {
  1250. return err
  1251. }
  1252. var belongData []*model.CustCustomerBelong
  1253. userName := s.GetCxtUserName()
  1254. for _, v := range rep {
  1255. orig_sale_name := v.SalesName
  1256. belong := new(model.CustCustomerBelong)
  1257. belong.CustId = v.Id
  1258. belong.SaleName = req.SalesName
  1259. belong.OrigSaleName = orig_sale_name
  1260. belong.OpnType = req.Receive
  1261. belong.OpnPeople = userName
  1262. belong.CreatedName = userName
  1263. belong.StartDate = gtime.Now() //新增开始时间
  1264. belong.OpnDatetime = gtime.Now()
  1265. belong.Remark = req.Remark
  1266. belong.CreatedBy = s.GetCxtUserId()
  1267. belongData = append(belongData, belong)
  1268. }
  1269. _, err = s.BelongDao.Insert(belongData)
  1270. return err
  1271. }
  1272. // 导出数据
  1273. func (s *CustomerService) Export(ctx context.Context, req *model.CustCustomerExport) (content *model.CustExport, err error) {
  1274. var con model.CustExport
  1275. req.IsRemovePage = true // 去掉分页标识
  1276. total, data, err := s.GetList(&req.CustCustomerSearchReq)
  1277. if err != nil {
  1278. return
  1279. }
  1280. cusType, err := service.GetDictDataByType(ctx, "cust_idy")
  1281. if err != nil {
  1282. return nil, err
  1283. }
  1284. f := excelize.NewFile()
  1285. index := f.NewSheet("Sheet1")
  1286. for index, item := range req.Columns {
  1287. sheetPosition := service.Div(index+1) + "1"
  1288. f.SetCellValue("Sheet1", sheetPosition, item)
  1289. }
  1290. if total > 0 {
  1291. for ck, item := range data {
  1292. for index, v := range req.Columns {
  1293. if v == "客户编码" {
  1294. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CustCode)
  1295. }
  1296. if v == "客户名称" {
  1297. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CustName)
  1298. }
  1299. if v == "助记名" {
  1300. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.AbbrName)
  1301. }
  1302. if v == "助记名" {
  1303. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.AbbrName)
  1304. }
  1305. if v == "所在地区" {
  1306. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CustLocation)
  1307. }
  1308. if v == "客户类型" {
  1309. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), cusType[item.CustIndustry])
  1310. }
  1311. if v == "客户级别" {
  1312. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CustLevel)
  1313. }
  1314. if v == "客户状态" {
  1315. var CustStatus string
  1316. CustStatus = "正常"
  1317. if item.CustStatus != "10" {
  1318. CustStatus = "异常"
  1319. }
  1320. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), CustStatus)
  1321. }
  1322. if v == "创建人" {
  1323. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CreatedName)
  1324. }
  1325. if v == "最后跟进时间" {
  1326. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.FollowUpDate)
  1327. }
  1328. if v == "创建时间" {
  1329. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CreatedTime)
  1330. }
  1331. if v == "所在省" {
  1332. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CustProvince)
  1333. }
  1334. if v == "所在市" {
  1335. f.SetCellValue("Sheet1", service.Div(index+1)+strconv.Itoa(ck+2), item.CustCity)
  1336. }
  1337. }
  1338. }
  1339. }
  1340. f.SetActiveSheet(index)
  1341. var buffer *bytes.Buffer
  1342. buffer, _ = f.WriteToBuffer()
  1343. con.Content = buffer.Bytes()
  1344. return &con, err
  1345. }