Index.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. <?php
  2. namespace app\client\controller;
  3. use app\event\FootPrints;
  4. use app\event\Msg;
  5. use app\model\Article;
  6. use app\model\Customer;
  7. use app\model\DesignerReserve;
  8. use app\model\Footprints as ModelFootprints;
  9. use app\model\MaterialCase;
  10. use app\model\MaterialEvidenceMedia;
  11. use app\model\MaterialEvidence;
  12. use app\model\Video;
  13. use app\model\Building;
  14. use app\model\Construction;
  15. use app\model\Recommend;
  16. use app\model\Footprints as FootprintsModel;
  17. use app\model\CustomerSubscribe;
  18. use app\model\Miniprogram;
  19. use app\model\Company;
  20. use app\model\CustomerClue;
  21. use app\model\Employee;
  22. use app\model\User;
  23. use app\model\AgentUser;
  24. use app\model\CustomerVisitLog;
  25. use think\facade\Db;
  26. use think\helper\Str;
  27. use app\logics\AgentsLogic;
  28. use app\model\TrainClass;
  29. use app\model\TrainCourse;
  30. use app\model\VrGroup;
  31. class Index extends Base
  32. {
  33. /**
  34. * 滚动报名信息展示
  35. */
  36. public function offer_data()
  37. {
  38. $data = Customer::withAttr('name', function ($value) {
  39. return mb_substr($value, 0, 1) . '老师';
  40. })->withAttr('phone', function ($value) {
  41. return substr_replace($value, '*******', 2, 7);
  42. })->field('name,phone')->limit(10)->order('addtime asc')->select();
  43. return json(['code' => 0, 'data' => $data]);
  44. }
  45. /**
  46. * 报名 视频下报名装修设计、热装楼盘询问装修、四个vr询问装修设计
  47. */
  48. public function signUp()
  49. {
  50. $token = request()->token;
  51. // 判断是否是员工
  52. if ($token['isEmployee']) return json(['code' => 1, 'msg' => '员工无需报名']);
  53. $employee_id = $token['share_employee'] ?? 0;
  54. $root_id = $token['root_org'];
  55. $data = $this->request->only(['name', 'remark', 'mobile']);
  56. $data = array_filter($data);
  57. $type = input('type', '', 'trim');
  58. $data_id = input('id', '', 'intval');
  59. $objData = [];
  60. switch ($type) {
  61. case 'video':
  62. $objData = Video::where('id', '=', $data_id)->findOrEmpty();
  63. break;
  64. case 'building':
  65. $objData = Building::where('id', '=', $data_id)->findOrEmpty();
  66. break;
  67. case 'materialCaseVr':
  68. $info = MaterialCase::where('id', '=', $data_id)->findOrEmpty();
  69. if (!$info->isEmpty()) {
  70. $objData['title'] = '案例"' . $info['title'] . '"VR';
  71. } else {
  72. $objData['title'] = '案例VR';
  73. }
  74. $objData['id'] = $data_id;
  75. break;
  76. case 'buildingVr':
  77. $info = Building::where('id', '=', $data_id)->findOrEmpty();
  78. if (!$info->isEmpty()) {
  79. $objData['title'] = '热装楼盘"' . $info['name'] . '"VR';
  80. } else {
  81. $objData['title'] = '热装楼盘VR';
  82. }
  83. $objData['id'] = $data_id;
  84. break;
  85. case 'constructionVr':
  86. $info = Construction::where('id', '=', $data_id)->findOrEmpty();
  87. if (!$info->isEmpty()) {
  88. $objData['title'] = '在施工地"' . $info['name'] . '"VR';
  89. } else {
  90. $objData['title'] = '在施工地VR';
  91. }
  92. $objData['id'] = $data_id;
  93. break;
  94. case 'companyVr':
  95. $objData['title'] = 'VR展厅';
  96. break;
  97. case 'employeeCardVr':
  98. $objData['title'] = 'VR名片';
  99. break;
  100. case 'TrainClassList':
  101. $info = Company::where('root_id', '=', $data_id)->findOrEmpty();
  102. if (!$info->isEmpty()) {
  103. $objData['title'] = $info['company_name'] . '的课程列表';
  104. } else {
  105. $objData['title'] = '课程列表';
  106. }
  107. //$objData['id'] = $data_id;
  108. break;
  109. case 'TrainClass':
  110. $info = TrainClass::where([['root_id', '=', $token['root_org']],['id','=',$data_id]])->findOrEmpty();
  111. if (!$info->isEmpty()) {
  112. $objData['title'] = $info['title'] . '的课程';
  113. } else {
  114. $objData['title'] = '学习课程';
  115. }
  116. $objData['id'] = $data_id;
  117. break;
  118. case 'TrainCourse':
  119. $info = TrainCourse::where([['root_id', '=', $token['root_org']],['id','=',$data_id]])->findOrEmpty();
  120. if (!$info->isEmpty()) {
  121. $objData['title'] = $info['title'] . '的课件';
  122. } else {
  123. $objData['title'] = '学习课件';
  124. }
  125. $objData['id'] = $data_id;
  126. break;
  127. default:
  128. break;
  129. }
  130. $data['data'] = $objData;
  131. $data['uid'] = $token['uid'];
  132. // 添加足迹
  133. if (!empty($data['uid'])) {
  134. event(new FootPrints($data['uid'], $employee_id, $root_id, $data, 'share'));
  135. }
  136. $user = User::findOrEmpty($data['uid']);
  137. if ($user->isEmpty()) {
  138. $user['nickname'] = '客户';
  139. }
  140. switch ($type) {
  141. case 'video':
  142. $new = $user['nickname'] . '预约了装修服务,请及时跟进';
  143. if ($employee_id) {
  144. event(new Msg($employee_id, $new, 'share_clue'));
  145. }
  146. break;
  147. case 'building':
  148. $new = $user['nickname'] . '询问了装修,请及时跟进';
  149. if ($employee_id) {
  150. event(new Msg($employee_id, $new, 'share_clue'));
  151. }
  152. break;
  153. default:
  154. $new = $user['nickname'] . '询问了设计,请及时跟进';
  155. if ($employee_id) {
  156. event(new Msg($employee_id, $new, 'share_clue'));
  157. }
  158. break;
  159. }
  160. return json(['code' => 0, 'msg' => '报名成功']);
  161. }
  162. /**
  163. * 获取装修设计方案报名
  164. */
  165. public function getDecorationDesign()
  166. {
  167. $token = request()->token;
  168. // 判断是否是员工
  169. if ($token['isEmployee']) return json(['code' => 1, 'msg' => '员工无需报名']);
  170. // 查询员工信息
  171. $employee_id = isset($token['share_employee']) ? intval($token['share_employee']) : 0;
  172. if (empty($employee_id)) return json(['code' => 0, 'msg' => '报名成功']);
  173. $employee = Employee::find($employee_id);
  174. $org_id = $employee->org_id;
  175. $root_id = $employee->root_id;
  176. $e_where[] = ['uid', '=', $token['uid']];
  177. $e_where[] = ['state', 'in', ['在职', '待审核']];
  178. $e_where[] = ['root_id', '=', $root_id];
  179. $isEmployee = Employee::where($e_where)->findOrEmpty();
  180. if (!$isEmployee->isEmpty()) return json(['code' => 1, 'msg' => '员工无需报名']);
  181. // 检测防止重复提交
  182. $hadList = ModelFootprints::where([['uid', '=', $token['uid']], ['employee_id', '=', $employee_id], ['pipe_type', '=', 'getDecorationDesign']])->count();
  183. if ($hadList) {
  184. return json(['code' => 1, 'msg' => '您已报名成功,请勿重复报名']);
  185. }
  186. $type = ucfirst(input('type', '', 'trim'));
  187. $id = input('id', '', 'intval');
  188. // 初始化类型
  189. $msgType = [
  190. 'Article' => ['拓客图文', 'title'],
  191. 'MaterialCase' => ['案例', 'title'],
  192. 'Building' => ['热装楼盘', 'name'],
  193. 'Construction' => ['在施工地', 'name'],
  194. 'MaterialEvidence' => ['客户好评', 'title'],
  195. 'Video' => ['拓客视频', 'title'],
  196. 'CompanyStrength' => ['公司实力', 'title']
  197. ];
  198. // 判断类型是否正确
  199. if (!isset($msgType[$type])) return json(['code' => 1, 'msg' => '类型不存在']);
  200. // 获取分享内容
  201. $class = 'app\\model\\' . $type;
  202. $obj = new $class();
  203. $objData = $obj->find($id);
  204. if (empty($objData)) return json(['code' => 1, 'msg' => '内容不存在']);
  205. // 获取报名信息
  206. $data = $this->request->only(['community', 'area', 'bedroom', 'mobile']);
  207. $data = array_filter($data);
  208. $data['data'] = $objData->toArray();
  209. $data['uid'] = $token['uid'];
  210. event(new FootPrints($data['uid'], $employee_id, $org_id, $data, 'getDecorationDesign'));
  211. $nickname = User::where('id', $token['uid'])->value('nickname');
  212. $msg = $nickname . '在' . $msgType[$type][0] . '《' . $objData[$msgType[$type][1]] . '》中申请获取装修方案,请及时处理';
  213. event(new Msg($employee_id, $msg, Str::snake($type)));
  214. // 更新线索电话
  215. $condition = [
  216. 'employee_id' => $employee_id,
  217. 'uid' => $token['uid'],
  218. 'org_id' => $org_id ?? 0
  219. ];
  220. // 建立线索关系
  221. $had = CustomerClue::where($condition)->find();
  222. if (empty($had->phone)) $had->phone = $data['mobile'];
  223. return json(['code' => 0, 'msg' => '报名成功']);
  224. }
  225. /**
  226. * 推荐内容(装修案例,图文素材,客户见证)
  227. */
  228. public function recommend($type, $page, $limit)
  229. {
  230. $order = $type . ' desc';
  231. $where['root_id'] = $this->request->token['root_org'];
  232. $where['publish'] = 1;
  233. $data = Recommend::with(['community', 'style'])->where($where)->order($order)->page($page, $limit)->select();
  234. foreach ($data as &$item) {
  235. $item->difference = 2; //1视频,2图文
  236. $item->cover = ''; //视频封面图
  237. switch ($item['type']) {
  238. case 'evidence':
  239. $arr = explode(',', $item->getData('cover_img'));
  240. if (count($arr) > 1) {
  241. $item['cover_img'] = $arr[0];
  242. }
  243. $type = MaterialEvidence::where('id', $item->id)->field('cover,difference')->find();
  244. $item->difference = $type->difference;
  245. $item->cover = $type->cover;
  246. $desc = MaterialEvidence::where('id', '=', $item['id'])->value('desc');
  247. $item->content = $desc;
  248. break;
  249. case 'article':
  250. $info = Article::where('id', '=', $item['id'])->withAttr('content', function ($value, $data) {
  251. $value = htmlspecialchars_decode($value);
  252. $value = preg_replace('/\s/', '', $value);
  253. $value = str_replace("&nbsp;", "", $value);
  254. $value = strip_tags($value);
  255. $text = mb_substr($value, 0, 40, "utf-8");
  256. return $text;
  257. })->field('content')->find();
  258. $item['content'] = $info['content'];
  259. break;
  260. case 'case':
  261. $info = MaterialCase::with(['designer'])->where('id', '=', $item['id'])->find();
  262. $item->vr_case = $info['vr_case'];
  263. if (empty($info->designer)) {
  264. $company = Company::where('root_id', '=', $info['root_id'])->find();
  265. $item->designer_name = $company['company_name'];
  266. $item->designer_headimgurl = $company['logo'];
  267. } else {
  268. $item->designer_name = $info->designer->name;
  269. if ($info['designer']['image_photo']) {
  270. $item->designer_headimgurl = $info['designer']['image_photo'];
  271. } else {
  272. $user_headimgurl = User::where('id', '=', $info->designer->uid)->value('headimgurl');
  273. $item->designer_headimgurl = $user_headimgurl;
  274. }
  275. }
  276. break;
  277. default:
  278. break;
  279. }
  280. }
  281. if (isset($this->request->token['share_employee'])) {
  282. event(new FootPrints($this->request->token['uid'], $this->request->token['share_employee'] ?? 0, $this->request->token['share_org'] ?? 0, [], 'weiwang'));
  283. }
  284. return json(['code' => self::success, 'data' => $data, 'msg' => '获取成功']);
  285. }
  286. /**
  287. * 修改浏览时长
  288. */
  289. public function visit_due_time()
  290. {
  291. $id = input('id');
  292. $type = input('pipe_type');
  293. $time = input('time');
  294. $token = $this->request->token;
  295. $type = $type=='housetype' ? 'buildingHousetype' : $type;
  296. $where[] = ['pipe_type', '=', $type];
  297. $where[] = ['uid', '=', $token['uid']];
  298. if (in_array($type, ['likeTest', 'styleTest', 'priceCalculation'])) { // 工具传值id为足迹ID
  299. FootprintsModel::where('id', $id)->update(['visit_due_time' => $time]);
  300. }elseif (in_array($type, ['groupVrVr','groupVr','group'])) {
  301. $id = VrGroup::where('id|sid',$id)->value('id');
  302. $where = [['uid', '=', $token['uid']],['pipe_type','in',['groupVrVr','groupVr','group']]];
  303. $id = FootprintsModel::where($where)->where('reg_info->"$.id" = ' . $id)->order('id desc')->value('id');
  304. if($id) FootprintsModel::where('id', $id)->update(['visit_due_time' => $time]);
  305. } else {
  306. if (in_array($type, ['card', 'employeeCardVr', 'toolAll'])) {
  307. $where[] = ['employee_id', '=', $id];
  308. $id = FootprintsModel::where($where)->order('id desc')->value('id');
  309. if ($id) {
  310. FootprintsModel::where('id', $id)->update(['visit_due_time' => $time]);
  311. }
  312. } else {
  313. $yid = $id;
  314. $id = FootprintsModel::where($where)->where('reg_info->"$.id" = ' . $id)->order('id desc')->value('id');
  315. if ($id) {
  316. FootprintsModel::where('id', $id)->update(['visit_due_time' => $time]);
  317. //增加记录素材的浏览时长到素材表
  318. if(!empty($time) && in_array($type,['article','materialEvidence','materialCase','CompanyStrength','video'])){
  319. $class = 'app\\model\\' . ucfirst($type);
  320. if (!class_exists($class)) return json(['code' => self::error_msg, 'msg' => '类型不存在']);
  321. $datas = (new $class())->where(['id' => $yid])->find();
  322. $old_due_time = $datas->visit_due_time;
  323. $datas->visit_due_time = $old_due_time + $time;
  324. $datas->save();
  325. }
  326. }
  327. }
  328. }
  329. return json(['code' => self::success, 'msg' => '操作成功']);
  330. }
  331. //预约
  332. public function subscribe()
  333. {
  334. $param = request()->param();
  335. $token = $this->request->token;
  336. $save['root_id'] = $token['root_org'];
  337. $save['sucai_id'] = $param['sucai_id'];
  338. $save['uid'] = $param['uid'];
  339. // $save['type'] = $param['type'];
  340. CustomerSubscribe::insert($save);
  341. return json(['code' => 0, 'msg' => '预约成功']);
  342. }
  343. /*
  344. * 分享人打开裂变内容,请求该内容需要展示的公司名称和logo
  345. */
  346. public function content_belong_company()
  347. {
  348. $miniType = $this->request->param('client_type');
  349. $uid = $this->request->param('uid');
  350. $root_id = Miniprogram::where('notify', $miniType)->value('root_id');
  351. $empid = Employee::where(['uid' => $uid, 'root_id' => $root_id])->value('id');
  352. $company = Company::where('root_id', $root_id)->field(['company_name', 'logo', 'company_address'])->find();
  353. return json(['code' => 0, 'msg' => '获取成功', 'data' => ['company_name' => $company['company_name'], 'logo' => $company['logo'], 'address' => $company['company_address'], 'employee_id' => $empid]]);
  354. }
  355. /*
  356. * 谈单工具各个板块足迹
  357. */
  358. public function toolAll()
  359. {
  360. $token = $this->request->token;
  361. $type = input('type',0);
  362. $arr = [0=>'',1=>'案例风格挑选',2=>'设计师推介',3=>'业主好评展示',4=>'在施工地展示',5=>'重点小区讲解',6=>'我的微官网',7=>'公司实力'];
  363. $nickname = User::where('id',$token['uid'])->value('nickname');
  364. $arr = ['uid'=>$token['uid'],'name'=>$nickname,'type'=>$arr[$type]];
  365. // 添加足迹
  366. if (!empty($token['uid']) && !$token['isEmployee']) {
  367. event(new FootPrints($token['uid'], $token['share_employee'] ?? 0, $token['share_org'] ?? 0, $arr, 'toolAll'));
  368. }
  369. return json(['code' => 0, 'msg' => '足迹添加成功']);
  370. }
  371. /**
  372. * 通过经纪人分享的内容,用户进行报名(小程序端)
  373. */
  374. public function add_customer()
  375. {
  376. $token = $this->request->token;
  377. $param = $this->request->only(['name', 'phone', 'community_name', 'agent_id']);
  378. $agtdata = AgentUser::where(['id' => $param['agent_id'], 'root_id' => $token['root_org']])->find();
  379. if (empty($agtdata)) return json(['code' => 1, 'msg' => '推荐人信息未查询到']);
  380. if ($token['uid'] == $agtdata['uid']) return json(['code' => 1, 'msg' => '您的身份是推荐人,不能报名']);
  381. if ($token['isEmployee']) return json(['code' => 1, 'msg' => '您已加入企业,不能参加活动']);
  382. // 添加客户检测手机号的重复
  383. $tips=(new AgentsLogic())->checkPepeat([$param['phone']], orgSubIds($agtdata['org_id']), $agtdata['agent_employee_id']);
  384. if(!empty($tips)) return $tips;
  385. $data = [
  386. 'employee_id' => $agtdata['agent_employee_id'],
  387. 'org_id' => $agtdata['org_id'],
  388. 'state' => 0,
  389. 'name' => $param['name'],
  390. 'phone' => cypherphone(trim($param['phone'])),
  391. 'community_name' => $param['community_name'],
  392. 'sex' => !empty($param['sex']) ? $param['sex'] : 1,
  393. 'agents_id' => $agtdata['id'],
  394. 'agent_source_type' => 3
  395. ];
  396. //保存报备的客户
  397. Db::startTrans();
  398. try {
  399. // 更新线索电话
  400. $condition = [
  401. 'employee_id' => $agtdata['agent_employee_id'],
  402. 'uid' => $token['uid'],
  403. 'org_id' => $agtdata['org_id']
  404. ];
  405. // 更新线索信息,并默认报备状态
  406. $had = CustomerClue::where($condition)->find();
  407. if (empty($had->phone)) {
  408. $had->phone = $param['phone'];
  409. $had->state = 1;
  410. $had->save();
  411. }
  412. $data['clue_id'] = $had['id'];
  413. $ms = Customer::insertGetId($data);
  414. $empuid = Employee::where('id', $agtdata['agent_employee_id'])->value('uid');
  415. $log = [
  416. 'employee_id' => $agtdata['agent_employee_id'],
  417. 'customer_id' => $ms,
  418. 'type' => '',
  419. 'remark' => '通过推荐人分享的内容报名的客户',
  420. 'user_id' => $empuid,
  421. 'state' => 1,
  422. 'org_id' => $agtdata['org_id']
  423. ];
  424. CustomerVisitLog::insertGetId($log);
  425. // 足迹
  426. if (!empty($token['uid']) && !$token['isEmployee']) {
  427. $footerArr = [
  428. 'name'=>$param['name'],
  429. 'phone'=>$param['phone'],
  430. 'community_name'=>$param['community_name'],
  431. ];
  432. event(new FootPrints($token['uid'], $agtdata['agent_employee_id'] ?? 0, $agtdata['org_id'] ?? 0, $footerArr, 'agentShareContentCrm'));
  433. }
  434. Db::commit();
  435. return json(['code' => 0, 'msg' => '报名成功']);
  436. } catch (\Exception $e) {
  437. Db::rollback();
  438. return json(['code' => 1, 'msg' => '报名失败']);
  439. }
  440. }
  441. }