Wenzhen.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604
  1. <?php
  2. namespace app\api\controller;
  3. use app\common\controller\Api;
  4. use think\Db;
  5. use addons\epay\library\Service;
  6. use app\common\library\Tenim;
  7. use app\common\model\Wenzhenorder;
  8. /**
  9. * 问诊订单
  10. */
  11. class Wenzhen extends Api
  12. {
  13. protected $noNeedLogin = '';
  14. protected $noNeedRight = '*';
  15. //创建订单
  16. public function create_order(){
  17. $member_id = input('member_id',0);
  18. $doctor_id = input('doctor_id',0);
  19. $ordertype = input('ordertype',0);
  20. $book_date = input('book_date',0); //2024-05-05
  21. $book_hour = input('book_time',0); //10:00
  22. $book_time = strtotime($book_date.' '.$book_hour);
  23. $member_disease = input('member_disease','');
  24. $member_images = input('member_images','');
  25. $auth_id = $this->auth->id;
  26. $order_no = createUniqueNo('W',$auth_id);
  27. //成员信息
  28. $member_info = Db::name('user_member')->where('user_id',$auth_id)->where('id',$member_id)->find();
  29. if(empty($member_info)){
  30. $this->error('不存在的就诊者');
  31. }
  32. //医生信息
  33. $doctor = Db::name('doctor')->where('id',$doctor_id)->find();
  34. $doctor_info = Db::name('doctor_info')->where('doctor_id',$doctor_id)->find();
  35. if(empty($doctor_info)){
  36. $this->error('不存在的医生');
  37. }
  38. //ordertype
  39. if(!in_array($ordertype,[1,2])){
  40. $this->error();
  41. }
  42. //价格
  43. $price = 0;
  44. if($ordertype == 1){
  45. if($doctor_info['typing_switch'] != 1){
  46. $this->error('该医生未开通图文问诊');
  47. }
  48. if($doctor['job_status'] != 1){
  49. $this->error('该医生现在不在上班中');
  50. }
  51. $price = $doctor_info['typing_price'];
  52. }
  53. if($ordertype == 2){
  54. if($doctor_info['video_switch'] != 1){
  55. $this->error('该医生未开通视频问诊');
  56. }
  57. if($doctor_info['video_model'] == 2 && $doctor['job_status'] != 1){ //即时方式,需要在上班中
  58. $this->error('该医生现在不在上班中');
  59. }
  60. $price = $doctor_info['video_price'];
  61. }
  62. if($price <= 0){
  63. $this->error('价格为零');
  64. }
  65. //排班判断
  66. if($ordertype == 1){
  67. $book_time = 0;
  68. }
  69. if($ordertype == 2){
  70. if($doctor_info['video_model'] == 1){ //排班预约
  71. //检查
  72. if($book_time < time() + 7200){
  73. //测试临时屏蔽
  74. //$this->error('最早只能预约'.date('m-d H:00',time() + 10800)); //最早预约两个小时后的一个整点
  75. }
  76. //排班状态
  77. $doctor_paiban = Db::name('doctor_paiban')->where('doctor_id',$doctor_id)->where('activetime',$book_time)->find();
  78. if(empty($doctor_paiban)){
  79. $this->error('您预约的时间该医生无法接诊');
  80. }
  81. if($doctor_paiban['active'] != 1){
  82. $this->error('您预约的时间该医生无法接诊');
  83. }
  84. //是否排满
  85. $map = [
  86. 'doctor_id' => $doctor_id,
  87. 'book_time' => $book_time,
  88. 'ordertype' => 2,
  89. 'status' => ['IN','10,20,25,30'],//有效订单
  90. ];
  91. $order_count = Db::name('wenzhen_order')->where($map)->count();
  92. if($order_count >= 4){
  93. $this->error('该时间点已排满,请重选预约时间');
  94. }
  95. }
  96. else
  97. {
  98. $book_time = time(); //即时预约
  99. }
  100. }
  101. //订单数据
  102. $order_data = [
  103. 'order_no' => $order_no,
  104. 'user_id' => $auth_id,
  105. 'member_id' => $member_id,
  106. 'doctor_id' => $doctor_id,
  107. 'ordertype' => $ordertype,
  108. 'total_fee' => $price,
  109. 'book_time' => $book_time,
  110. 'createtime' => time(),
  111. 'status' => 0,//订单状态enum
  112. 'video_model'=> $doctor_info['video_model'], //视频问诊预约方式
  113. ];
  114. Db::startTrans();
  115. //下单
  116. $order_id = Db::name('wenzhen_order')->insertGetId($order_data);
  117. if(!$order_id){
  118. Db::rollback();
  119. $this->error('下单失败');
  120. }
  121. //订单附表,就诊人信息
  122. $member_data = $member_info;
  123. unset($member_data['id']);
  124. $member_data['order_id'] = $order_id;
  125. $member_data['member_id'] = $member_info['id'];
  126. $member_data['member_disease'] = $member_disease;
  127. $member_data['member_images'] = $member_images;
  128. $order_member_id = Db::name('wenzhen_order_member')->insertGetId($member_data);
  129. if(!$order_member_id){
  130. Db::rollback();
  131. $this->error('下单失败');
  132. }
  133. //
  134. Db::commit();
  135. $this->success('下单成功',$order_id);
  136. }
  137. //支付订单
  138. public function pay_order(){
  139. $order_id = input('order_id',0);
  140. $pay_type = input('pay_type','wechat');
  141. //pay_type
  142. if(!in_array($pay_type,['wechat','alipay','wallet'])){
  143. $this->error();
  144. }
  145. //订单详情
  146. $wenzhen_order = Db::name('wenzhen_order')->where('user_id',$this->auth->id)->where('id',$order_id)->find();
  147. if(empty($wenzhen_order)){
  148. $this->error('不存在的订单');
  149. }
  150. if($wenzhen_order['status'] != 0){
  151. $this->error('该订单不能支付');
  152. }
  153. if($pay_type == 'wallet'){
  154. Db::startTrans();
  155. //用户扣钱
  156. $logtype = $wenzhen_order['ordertype'] == 1 ? 11 : 12;
  157. $rs_wallet = model('wallet')->lockChangeAccountRemain($this->auth->id,'money',-$wenzhen_order['total_fee'],$logtype,'图文问诊'.$wenzhen_order['order_no'],'wenzhen_order',$order_id);
  158. if($rs_wallet['status'] == false){
  159. Db::rollback();
  160. $this->error($rs_wallet['msg']);
  161. }
  162. //订单改为已支付
  163. $wenzhen_update['status'] = 10;//订单状态enum
  164. $wenzhen_update['pay_type'] = $pay_type;
  165. $wenzhen_update['pay_time'] = time();
  166. $update_rs = Db::name('wenzhen_order')->where('id',$order_id)->where('status',0)->update($wenzhen_update);
  167. if(!$update_rs){
  168. Db::rollback();
  169. $this->error('支付失败,重试一下吧');
  170. }
  171. Db::commit();
  172. //发送im消息给医生
  173. if($wenzhen_order['ordertype'] == 1){
  174. $tenim = new Tenim();
  175. $message = [
  176. 'businessID' => 'order_status',
  177. 'name' => '待接单',
  178. 'status' => '10',
  179. 'id' => (string)$order_id,
  180. 'content' => '已通知医生尽快接诊,超时未接诊的将自动取消订单并退款',
  181. ];
  182. $rs = $tenim->sendCustomMessageToUser('user'.$this->auth->id,'doctor'.$wenzhen_order['doctor_id'],$message);
  183. }
  184. //输出
  185. $result = [
  186. 'pay_type' => $pay_type,
  187. 'pay_params' => '',
  188. ];
  189. $this->success(1,$result);
  190. }
  191. //创建订单
  192. $platform = 'app';
  193. $data = [];
  194. $data['user_id'] = $this->auth->id;
  195. $data['out_trade_no'] = createUniqueNo('W',$this->auth->id);
  196. $data['order_amount'] = $wenzhen_order['total_fee'];
  197. $data['createtime'] = time();
  198. $data['pay_type'] = $pay_type;
  199. $data['platform'] = $platform;
  200. $data['order_status'] = 0;
  201. $data['table_name'] = 'wenzhen_order';
  202. $data['table_id'] = $order_id;
  203. $data['args'] = '';
  204. $orderid = Db::name('pay_order')->insertGetId($data);
  205. $openid = '';
  206. //下单
  207. $params = [
  208. 'type' => $pay_type,
  209. 'orderid' => $data['out_trade_no'],
  210. 'title' => '支付订单',
  211. 'amount' => $data['order_amount'],
  212. 'method' => $platform,
  213. 'openid' => $openid,
  214. 'notifyurl' => config('pay_notify_url').'/api/notify/wenzhen_notify_base/paytype/'.$pay_type,
  215. 'returnurl' => '',
  216. ];
  217. $res = Service::submitOrder($params);
  218. $result = [
  219. 'pay_type'=> $pay_type,
  220. 'pay_params'=> $pay_type
  221. ];
  222. if($pay_type == 'wechat'){
  223. $result['pay_params'] = json_decode($res,true);
  224. }else{
  225. $result['pay_params'] = $res;
  226. }
  227. $this->success(1,$result);
  228. }
  229. //订单列表
  230. public function order_list(){
  231. $status = input('status','all');
  232. $where = ['user_id'=>$this->auth->id];
  233. if($status != 'all'){
  234. $where['order.status'] = $status;
  235. if($status == 20){// 已接诊、问诊中
  236. $where['order.status'] = ['IN',[20,25]];
  237. }
  238. }
  239. $field = [
  240. 'order.id','order.order_no','order.createtime','order.status','order.ordertype','order.accept_time','order.video_time','order.doctor_id','order.cancel_reason','order.comefrom',
  241. 'doctor.avatar as doctor_avatar','doctor.nickname as doctor_nickname',
  242. ];
  243. $list = Db::name('wenzhen_order')->alias('order')
  244. ->field($field)
  245. ->join('doctor','order.doctor_id = doctor.id','LEFT')
  246. ->order('order.id desc')
  247. ->where($where)
  248. ->autopage()->select();
  249. $list = list_domain_image($list,['doctor_avatar']);
  250. if(!empty($list)){
  251. //循环处理
  252. foreach($list as $key => $val){
  253. $val = $this->orderinfo_appen($val);
  254. //医生原因的退款,都直接算到退珍
  255. if(in_array($val['status'],[16,18,22])){
  256. $val['status'] = 100;
  257. }
  258. //视频拨打中 也算问诊中
  259. if($val['status'] == 25){
  260. $val['status'] = 20;
  261. }
  262. $list[$key] = $val;
  263. }
  264. }
  265. $this->success(1,$list);
  266. }
  267. //单个订单,追加数据
  268. private function orderinfo_appen($val){
  269. //订单类型
  270. $ordertype_data = [
  271. 1 => ['name'=>'图文订单','name_en'=>'Visual Order'],
  272. 2 => ['name'=>'视频订单','name_en'=>'Video Order'],
  273. ];
  274. //状态备注
  275. $status_data = $this->order_status_enum();
  276. //开始处理
  277. //订单类型,辅助
  278. $val['ordertype_data'] = $ordertype_data[$val['ordertype']];
  279. //订单状态,辅助
  280. $status_data_val = $status_data[$val['status']];
  281. //未支付订单N分钟后自动取消
  282. if($val['status'] == 0){
  283. $remain_time = $val['createtime'] + (config('site.nopay_order_autocancel_minute')*60) - time();
  284. if($remain_time > 0){
  285. $status_data_val['minute'] = Sec2Time($remain_time);
  286. $status_data_val['minute_en'] = Sec2Time_en($remain_time);
  287. $status_data_val['seconds'] = $remain_time;
  288. }
  289. }
  290. //被医生退珍,追加原因
  291. if($val['status'] == 18){
  292. $status_data_val['name'] .= ':'.$val['cancel_reason'];
  293. }
  294. //图文订单接诊后N分钟自动结束
  295. if($val['status'] == 20 && $val['ordertype'] == 1){
  296. $remain_time = $val['accept_time'] + (config('site.accept_textorder_autofinish_minute')*60) - time();
  297. if($remain_time > 0){
  298. $status_data_val['minute'] = Sec2Time($remain_time);
  299. $status_data_val['minute_en'] = Sec2Time_en($remain_time);
  300. $status_data_val['seconds'] = $remain_time;
  301. }
  302. }
  303. //视频订单接通后N分钟自动结束
  304. if($val['status'] == 25 && $val['ordertype'] == 2){
  305. $minute = config('site.firstvideo_videoorder_autofinish_minute');
  306. if($val['comefrom'] == 2){
  307. $minute = config('site.tv_firstvideo_videoorder_autofinish_minute');
  308. }
  309. $remain_time = $val['video_time'] + ($minute*60) - time();
  310. if($remain_time > 0){
  311. $status_data_val['minute'] = Sec2Time($remain_time);
  312. $status_data_val['minute_en'] = Sec2Time_en($remain_time);
  313. $status_data_val['seconds'] = $remain_time;
  314. }
  315. }
  316. $val['status_data'] = $status_data_val;
  317. //处理完成,返回
  318. return $val;
  319. }
  320. //订单状态备注枚举
  321. private function order_status_enum(){
  322. //状态:0=未付款,3=已失效,10=待接诊,13=用户退诊,16=医生未接诊,18=医生退诊,20=已接诊,22=超时未拨打,25=视频通话中,30=问诊完成
  323. $status_data = [
  324. 0 => [
  325. 'name'=>'等待支付时间',
  326. 'name_en'=>'Waiting time for payment',
  327. 'minute' => '',
  328. 'minute_en' => '',
  329. 'seconds' => 0,
  330. ],
  331. 3 => [
  332. 'name'=>'您的订单未及时支付,已失效',
  333. 'name_en'=>'Your order was not paid on time and has expired.',
  334. 'minute' => '',
  335. 'minute_en' => '',
  336. 'seconds' => 0,
  337. ],
  338. 10 => [
  339. 'name'=>'等待医生接诊:已通知医生尽快接诊,超时自动取消订单并退款',
  340. 'name_en'=>'Waiting for the doctor. If not accepted within the specified time,the consultation will be automatically canceled, and a refund will be issued',
  341. 'minute' => '',
  342. 'minute_en' => '',
  343. 'seconds' => 0,
  344. ],
  345. 13 => [
  346. 'name'=>'您的订单已退珍',
  347. 'name_en'=>'Your order has been canceled',
  348. 'minute' => '',
  349. 'minute_en' => '',
  350. 'seconds' => 0,
  351. ],
  352. 16 => [
  353. 'name'=>'医生未接诊',
  354. 'name_en'=>'Doctor Not received',
  355. 'minute' => '',
  356. 'minute_en' => '',
  357. 'seconds' => 0,
  358. ],
  359. 18 => [
  360. 'name'=>'退珍原因',
  361. 'name_en'=>'',
  362. 'minute' => '',
  363. 'minute_en' => '',
  364. 'seconds' => 0,
  365. ],
  366. 20 => [
  367. 'name'=>'医生已接诊,咨询中',
  368. 'name_en'=>'',
  369. 'minute' => '',
  370. 'minute_en' => '',
  371. 'seconds' => 0,
  372. ],
  373. 22 => [
  374. 'name'=>'医生超时未拨打视频',
  375. 'name_en'=>'',
  376. 'minute' => '',
  377. 'minute_en' => '',
  378. 'seconds' => 0,
  379. ],
  380. 25 => [
  381. 'name'=>'视频问诊进行中',
  382. 'name_en'=>'Time Remaining',
  383. 'minute' => '',
  384. 'minute_en' => '',
  385. 'seconds' => 0,
  386. ],
  387. 30 => [
  388. 'name'=>'点击查看医生反馈结果',
  389. 'name_en'=>'Click to view the consultation results',
  390. 'minute' => '',
  391. 'minute_en' => '',
  392. 'seconds' => 0,
  393. ],
  394. ];
  395. return $status_data;
  396. }
  397. //订单详情
  398. public function order_info(){
  399. $order_id = input('order_id',0);
  400. //订单详情
  401. $wenzhen_order = Db::name('wenzhen_order')->where('user_id',$this->auth->id)->where('id',$order_id)->find();
  402. if(empty($wenzhen_order)){
  403. $this->error('不存在的订单');
  404. }
  405. $wenzhen_order = info_domain_image($wenzhen_order,['feedback_images']);
  406. $wenzhen_order = $this->orderinfo_appen($wenzhen_order);
  407. //就诊人详情
  408. $order_member_info = Db::name('wenzhen_order_member')->where('order_id',$order_id)->find();
  409. $order_member_info = info_domain_image($order_member_info,['member_images']);
  410. $order_member_info['age'] = birthtime_to_age($order_member_info['birthday']);
  411. $wenzhen_order['member_info'] = $order_member_info;
  412. //医生详情
  413. $field = [
  414. 'd.nickname','d.avatar','d.keshi_id','d.level_id','d.hospital','d.goodat','d.ordernum',
  415. 'keshi.name as keshi_name',
  416. 'level.name as level_name'
  417. ];
  418. $doctor_info = Db::name('doctor')->alias('d')
  419. ->field($field)
  420. ->join('doctor_level level','d.level_id = level.id','LEFT')
  421. ->join('keshi','d.keshi_id = keshi.id','LEFT')
  422. ->where('d.id',$wenzhen_order['doctor_id'])->find();
  423. $doctor_info = info_domain_image($doctor_info,['avatar']);
  424. $wenzhen_order['doctor_info'] = $doctor_info;
  425. $this->success(1,$wenzhen_order);
  426. }
  427. //编辑订单里的就诊人资料
  428. public function edit_member(){
  429. $order_id = input('order_id',0);
  430. //订单详情
  431. $wenzhen_order_member = Db::name('wenzhen_order_member')->where('user_id',$this->auth->id)->where('order_id',$order_id)->find();
  432. if(empty($wenzhen_order_member)){
  433. $this->error('不存在的订单');
  434. }
  435. $update = [
  436. 'member_disease' => input('member_disease',''),
  437. 'member_images' => input('member_images',''),
  438. ];
  439. Db::name('wenzhen_order_member')->where('user_id',$this->auth->id)->where('order_id',$order_id)->update($update);
  440. $this->success();
  441. }
  442. //用户发起退珍
  443. //因为存在第三方,悲观锁
  444. public function tuizhen(){
  445. $apilimit = $this->apiLimit();
  446. if(!$apilimit){
  447. $this->error('操作频繁');
  448. }
  449. //订单详情
  450. $order_id = input('order_id',0);
  451. Db::startTrans();
  452. $wenzhen_order = Db::name('wenzhen_order')->where('user_id',$this->auth->id)->where('id',$order_id)->lock(true)->find();
  453. if(empty($wenzhen_order)){
  454. Db::rollback();
  455. $this->error('不存在的订单');
  456. }
  457. //仅限视频订单
  458. if($wenzhen_order['ordertype'] == 1){
  459. Db::rollback();
  460. $this->error('图文问诊不能退诊');
  461. }
  462. //状态限制
  463. if(!in_array($wenzhen_order['status'],[10,20])){
  464. Db::rollback();
  465. $this->error('当前订单不能退诊');
  466. }
  467. //超过支付时间一小时不能退
  468. if( time() - $wenzhen_order['pay_time'] > 3600){
  469. Db::rollback();
  470. $this->error('支付超过一小时,不能退诊');
  471. }
  472. //修改订单
  473. $nowtime = time();
  474. $update = [
  475. 'status' => 13,
  476. 'cancel_time' => $nowtime,
  477. 'cancel_reason' => '用户付款后退诊',
  478. 'finish_time' => $nowtime,
  479. ];
  480. //退款要打折扣
  481. $wenzhen_refund_bili = config('site.payorder_userrefund_bili') ?: 70;
  482. $refund_price = bcdiv(bcmul($wenzhen_order['total_fee'],$wenzhen_refund_bili,2),100,2);
  483. //余额支付的直接退款
  484. if($wenzhen_order['pay_type'] == 'wallet'){
  485. //用户加钱
  486. $logtype = $wenzhen_order['ordertype'] == 1 ? 13 : 14;
  487. $rs_wallet = model('wallet')->lockChangeAccountRemain($wenzhen_order['user_id'],'money',$refund_price,$logtype,$remark='用户问诊退珍','wenzhen_order',$order_id);
  488. if($rs_wallet['status'] === false){
  489. Db::rollback();
  490. $this->error($rs_wallet['msg']);
  491. }
  492. //订单退款参数
  493. $update['refund_price'] = $refund_price;
  494. $update['refund_status'] = 3;
  495. }else{
  496. $update['refund_status'] = 1; //待退款
  497. //发起退款
  498. $Wenzhenorder_model = new Wenzhenorder;
  499. $refund_rs = $Wenzhenorder_model->old_refund($wenzhen_order,$refund_price);
  500. if($refund_rs === true){
  501. $update['refund_status'] = 3; //退款完成
  502. }
  503. }
  504. $update_rs = Db::name('wenzhen_order')->where('id',$order_id)->update($update);
  505. if(!$update_rs){
  506. Db::rollback();
  507. $this->error('退诊失败');
  508. }
  509. Db::commit();
  510. $this->success('退珍成功');
  511. }
  512. }