Order.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. <?php
  2. /**
  3. * Created by PhpStorm.
  4. * User: zilongs
  5. * Date: 20-9-30
  6. * Time: 下午10:56
  7. */
  8. namespace App\Models;
  9. class Order extends BaseModel
  10. {
  11. protected $appends = ['is_evaluate', 'consult_duration', 'callback_phone'];
  12. CONST UNPAID = 1, NOTACCEPT = 2, ISING = 3, FINISHED = 4,CANCELED=5,ISOUT=6; //订单状态(1.未支付 2.进行中 3.已完成 4.已取消)
  13. public static $_order_status = [
  14. self::UNPAID=>'未支付',
  15. self::NOTACCEPT=>'待支付',
  16. self::ISING=>'进行中',
  17. self::FINISHED=>'已完成',
  18. self::CANCELED=>'已取消',
  19. self::ISOUT=>'已超时'
  20. ];
  21. //获取订单状态
  22. public static function getStatus()
  23. {
  24. return self::$_order_status;
  25. }
  26. public function docter()
  27. {
  28. return $this->belongsTo(Docter::class);
  29. }
  30. public function calllog()
  31. {
  32. return $this->hasMany(CallLog::class);
  33. }
  34. /**
  35. * 用户医生关注表
  36. * @return \Illuminate\Database\Eloquent\Relations\BelongsTo
  37. * @author Liu-Yh
  38. * Create By 2020/11/18 11:06
  39. */
  40. public function userDocter(){
  41. return $this->hasOne(UserDocter::class,'user_id','user_id')->select(['id', 'remark']);
  42. }
  43. public function patients(){
  44. return $this->belongsTo(Patient::class);
  45. }
  46. public function orderPatient()
  47. {
  48. return $this->hasOne(OrderPatient::class);
  49. }
  50. public function orderPack()
  51. {
  52. return $this->hasOne(OrderPack::class);
  53. }
  54. public function orderNurse()
  55. {
  56. return $this->hasMany(OrderNurse::class);
  57. }
  58. public function orderVaccine()
  59. {
  60. return $this->hasOne(OrderVaccine::class);
  61. }
  62. public function orderUser()
  63. {
  64. return $this->hasOne(User::class,'id','user_id');
  65. }
  66. public function organization()
  67. {
  68. return $this->belongsTo(Organization::class);
  69. }
  70. public function user()
  71. {
  72. return $this->belongsTo(User::class);
  73. }
  74. public function evaluate()
  75. {
  76. return $this->hasOne(Evaluate::class);
  77. }
  78. //支付完成的处理方法
  79. public static function payCompletedHandle($order_id)
  80. {
  81. $order = Order::with(['organization', 'orderVaccine', 'orderNurse', 'orderPack'])->select(['user_id', 'docter_id', 'product_type', 'total_amount', 'payment_type', 'payment_amount', 'order_sn', 'pay_order_pack_id', 'organization_id'])->where('id', $order_id)->first();
  82. $orderPatient = OrderPatient::where('order_id', $order_id)->first();
  83. $user = User::select(['balance', 'openid'])->where('id', $order['user_id'])->first();
  84. //发送下单消息
  85. if ($order['product_type'] < 6) {
  86. $product_type_text = config('config.product_type_map')[$order['product_type']];
  87. UserMessage::saveMessage($order['user_id'], 4, $order_id, [$product_type_text]);
  88. }
  89. elseif ($order['product_type'] == 6) {
  90. $orderPack = OrderPack::select(['pack_name', 'end_time'])->where('order_id', $order_id)->first();
  91. UserMessage::saveMessage($order['user_id'], 5, $order_id, [$orderPack['pack_name'], date('Y-m-d', $orderPack['end_time'])]);
  92. //更新用户为服务包用户
  93. User::where('id', $order['user_id'])->update(['is_pack' => 1]);
  94. }
  95. elseif ($order['product_type'] == 7) {
  96. UserMessage::saveMessage($order['user_id'], 7, $order_id, [round($order['total_amount']/100, 2), round($user['balance']/100, 2)]);
  97. }
  98. //发送余额付款成功消息
  99. if ($order['payment_type'] == 2) {
  100. UserMessage::saveMessage($order['user_id'], 8, $order_id, [round($order['payment_amount']/100, 2), round($user['balance']/100, 2)]);
  101. }
  102. //发送医生端消息
  103. DocterMessage::saveMessage($order['docter_id'], $order['user_id'], 1, $order_id, [$order['order_sn']]);
  104. //如果是服务包支付的,就扣服务包的次数
  105. if ($order['payment_type'] == 3) {
  106. OrderPack::deductPackData($order['pay_order_pack_id'], $order['product_type']);
  107. }
  108. //如果是门诊预约,预约时间段的订单数要加1
  109. if ($order['product_type'] == 3) {
  110. $schedule_date = date('Y-m-d', $orderPatient['appoint_start_time']);
  111. SchedulePeriod::where('docter_id', $order['docter_id'])->where('organization_id', $order['organization_id'])->where('schedule_type', 1)->where('time_period_id', $orderPatient['time_period_id'])->where('schedule_date', $schedule_date)->increment('order_num');
  112. }
  113. //如果是儿保和疫苗预约,预约时间段的订单数要加1
  114. if (in_array($order['product_type'], [4,5])) {
  115. $schedule_date = date('Y-m-d', $orderPatient['appoint_start_time']);
  116. $schedule_type = $order['product_type'] == 4 ? 2 : 3;
  117. SchedulePeriod::where('organization_id', $order['organization_id'])->where('schedule_type', $schedule_type)->where('time_period_id', $orderPatient['time_period_id'])->where('schedule_date', $schedule_date)->increment('order_num');
  118. }
  119. //如果是疫苗就减少疫苗库存
  120. if ($order['product_type'] == 4) {
  121. $orderVaccine = OrderVaccine::where('order_id', $order_id)->first();
  122. OrganizationVaccine::where('org_id', $order['organization_id'])->where('vaccine_id', $orderVaccine['vaccine_id'])->decrement('stock');
  123. }
  124. //发送微信消息
  125. $docter = Docter::where('id', $order['docter_id'])->first();
  126. if ($order['product_type'] == 1) {
  127. $official_arr = [$user['openid'], $docter['name'], date('Y-m-d H:i:s'), round($order['payment_amount']/100, 2)];
  128. $subscribe_arr = [];
  129. send_wechat_message(1, $official_arr, $subscribe_arr);
  130. }
  131. if ($order['product_type'] == 2) {
  132. $official_arr = [$user['openid'], $docter['name'], $orderPatient['name'], $orderPatient['symptoms'], date('Y-m-d H:i:s'), round($order['payment_amount']/100, 2)];
  133. $subscribe_arr = [];
  134. send_wechat_message(2, $official_arr, $subscribe_arr);
  135. }
  136. if ($order['product_type'] == 3) {
  137. $keyword2 = date('Y-m-d H:i', $orderPatient['appoint_start_time']).' - '.date('H:i', $orderPatient['appoint_end_time']);
  138. $official_arr = [$user['openid'], $docter['name'], $keyword2, $orderPatient['name'], $orderPatient['phone'], $order['organization']['name'], $order['organization']['address']];
  139. $subscribe_arr = [];
  140. send_wechat_message(3, $official_arr, $subscribe_arr);
  141. }
  142. if ($order['product_type'] == 4) {
  143. $keyword2 = date('Y-m-d H:i', $orderPatient['appoint_start_time']).' - '.date('H:i', $orderPatient['appoint_end_time']);
  144. $official_arr = [$user['openid'], $order['organization']['name'], $order['order_vaccine']['vaccine_name'], $keyword2, $orderPatient['name'], $orderPatient['phone'], $order['organization']['name'], $order['organization']['address']];
  145. $subscribe_arr = [];
  146. send_wechat_message(4, $official_arr, $subscribe_arr);
  147. }
  148. if ($order['product_type'] == 5) {
  149. $keyword2 = date('Y-m-d H:i', $orderPatient['appoint_start_time']).' - '.date('H:i', $orderPatient['appoint_end_time']);
  150. $official_arr = [$user['openid'], $order['organization']['name'], $order['order_nurse'][0]['nurse_name'], $keyword2, $orderPatient['name'], $orderPatient['phone'], $order['organization']['name'], $order['organization']['address']];
  151. $subscribe_arr = [];
  152. send_wechat_message(5, $official_arr, $subscribe_arr);
  153. }
  154. if ($order['product_type'] == 6) {
  155. $official_arr = [$user['openid'], $order['order_pack']['name'], date('Y-m-d H:i:s', $order['order_pack']['start_time']), date('Y-m-d H:i:s', $order['order_pack']['end_time'])];
  156. $subscribe_arr = [];
  157. send_wechat_message(6, $official_arr, $subscribe_arr);
  158. }
  159. return true;
  160. }
  161. public function suggest()
  162. {
  163. return $this->hasOne(Suggest::class)->select(['id', 'order_id']);
  164. }
  165. public function getIsEvaluateAttribute()
  166. {
  167. $is_evaluate = 0;
  168. $user = User::getUserByToken(false);
  169. if (!empty($user)) {
  170. if (Evaluate::where('order_id', $this->id)->where('user_id', $user['id'])->exists()) {
  171. $is_evaluate = 1;
  172. }
  173. }
  174. return $is_evaluate;
  175. }
  176. //取消订单,建议是在事务里面去调用
  177. public static function orderCancel($order_id, $order_notes = '')
  178. {
  179. $order = Order::with(['orderPatient'])->where('id', $order_id)->first();
  180. //判断预约时间是否还剩1小时内
  181. $can_refund = true;
  182. if (in_array($order['product_type'], [3,4,5])) {
  183. $order_notes = '用户取消预约';
  184. if ($order['order_patient']['appoint_start_time'] - 3600 < time()) {
  185. $order_notes = '用户超时取消';
  186. $can_refund = false;
  187. }
  188. }
  189. //改变订单状态
  190. $updateOrder = ['order_status' => 5, 'order_notes' => $order_notes];
  191. if ($order['payment_status'] > 1) {
  192. //退钱到余额
  193. if (!empty($order['payment_amount']) && $can_refund && $order['product_type'] != 5) {
  194. User::changeBalance($order_id, $order['payment_amount'], 4, $order['id'], '取消订单退款');
  195. }
  196. $updateOrder['cancel_time'] = time();
  197. if ($can_refund) {
  198. $updateOrder['payment_status'] = 4;
  199. }
  200. if ($order['product_type'] == 5) {
  201. $updateOrder['payment_status'] = 5;
  202. }
  203. }
  204. Order::where('id', $order_id)->update($updateOrder);
  205. //如果是门诊预约且距离预约时间还有1个小时以上,那么预约时间段的订单数减1
  206. if ($order['product_type'] == 3 && $can_refund) {
  207. $schedule_date = date('Y-m-d', $order['order_patient']['appoint_start_time']);
  208. SchedulePeriod::where('docter_id', $order['docter_id'])->where('organization_id', $order['organization_id'])->where('time_period_id', $order['order_patient']['time_period_id'])->where('schedule_date', $schedule_date)->decrement('order_num');
  209. }
  210. //如果是疫苗儿保预约且距离预约时间还有1个小时以上,那么预约时间段的订单数减1
  211. if (in_array($order['product_type'], [4,5]) && $can_refund) {
  212. $schedule_type_map = [4 => 2, 5 => 3];
  213. $schedule_date = date('Y-m-d', $order['order_patient']['appoint_start_time']);
  214. SchedulePeriod::where('organization_id', $order['organization_id'])->where('time_period_id', $order['order_patient']['time_period_id'])->where('schedule_date', $schedule_date)->where('schedule_type', $schedule_type_map[$order['product_type']])->decrement('order_num');
  215. }
  216. //如果是疫苗预约那么取消订单就增加疫苗库存
  217. if ($order['product_type'] == 4) {
  218. $orderVaccine = OrderVaccine::where('order_id', $order_id)->first();
  219. OrganizationVaccine::where('org_id', $order['organization_id'])->where('vaccine_id', $orderVaccine['vaccine_id'])->increment('stock');
  220. }
  221. return true;
  222. }
  223. public function getConsultDurationAttribute()
  224. {
  225. $duration = 0;
  226. if ($this->product_type == 1) {
  227. $duration = CallLog::where('order_id', $this->id)->where('talk_time', '<>', null)->sum('talk_time');
  228. $duration = !empty($duration) ? $duration : 0;
  229. }
  230. elseif ($this->product_type == 2) {
  231. $duration = $this->end_time - $this->receiving_time;
  232. }
  233. return $duration > 0 ? $duration : 0;
  234. }
  235. public function getCallbackPhoneAttribute()
  236. {
  237. $secret_no = '';
  238. if ($this->product_type == 1) {
  239. $secret_no = CallLog::where('order_id', $this->id)->orderBy('id', 'desc')->value('secret_no');
  240. }
  241. return !empty($secret_no) ? $secret_no : '';
  242. }
  243. }