cust_customer.go 43 KB

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