UserOrder.php 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805
  1. <?php
  2. namespace app\common\model;
  3. use app\admin\model\Admin;
  4. use app\service\pay_user\PayUser;
  5. use app\service\system\OrderConfig;
  6. use think\Db;
  7. use think\db\Query;
  8. use think\Model;
  9. use think\model\Collection;
  10. use traits\model\SoftDelete;
  11. /**
  12. *@property array send_detail 运送详情
  13. *@property float cal_refund 退款手续费
  14. *@property float refund_money 退款金额
  15. *@property Collection currentSenderOrder
  16. *@property float settle_amount
  17. *@property array senders
  18. *@property array settle_info
  19. *@property float profit
  20. *@property boolean is_same_user
  21. *@property User user
  22. *@method Query|self expired()
  23. *@method Query|self show()
  24. *@method Query|self service()()
  25. *@method Query|self waitGet()
  26. *@method Query|self running($from='')
  27. *@method Query|self success()
  28. *@method Query|self settled()
  29. *@method Query|self settle()
  30. *@method Query|self canceled()
  31. *@method Query|self completed()
  32. *@method Query|self caled()
  33. *@method Query|self profit($data)
  34. *@method Query|self over()
  35. *@method Query|self turnover($date='today')
  36. */
  37. class UserOrder extends Model
  38. {
  39. //use SoftDelete;
  40. protected $hidden=[
  41. 'status_bak',
  42. 'deleted_at',
  43. ];
  44. protected $autoWriteTimestamp=true;
  45. protected $createTime = 'created_at';
  46. protected $updateTime = 'updated_at';
  47. protected $deleteTime='deleted_at';
  48. protected $type=[
  49. 'has_cage'=>'boolean',
  50. 'from_addr'=>'json',
  51. 'to_addr'=>'json',
  52. 'refund_images'=>'array',
  53. ];
  54. const FREIGHT_FAST='fast';#快车
  55. const FREIGHT_AIR='air';#空运
  56. const FREIGHT_SPECIAL='special';#专车
  57. public static $freights=[
  58. self::FREIGHT_FAST=>'快车',
  59. self::FREIGHT_AIR=>'空运',
  60. self::FREIGHT_SPECIAL=>'专车',
  61. ];
  62. const STATUS_WAIT_PAY=0;
  63. const STATUS_WAIT_GET=1;
  64. const STATUS_GETTING=10;
  65. const STATUS_SENDING=20;
  66. const STATUS_GIVING=30;
  67. const STATUS_GIVED=40;
  68. const STATUS_COMPLETE=50;
  69. const STATUS_WAIT_SET=55;
  70. const STATUS_SET=60;
  71. const STATUS_SET_REJECT=65;
  72. const STATUS_REFUND=70;
  73. const STATUS_CANCEL=80;
  74. public static $statusList=[
  75. self::STATUS_WAIT_PAY=>'待支付',
  76. self::STATUS_WAIT_GET=>'待接单',
  77. self::STATUS_GETTING=>'已接单,取宠中',
  78. self::STATUS_SENDING=>'已取宠,运输中',
  79. self::STATUS_GIVING=>'送宠中',
  80. self::STATUS_GIVED=>'送宠完成,待用户确认',
  81. self::STATUS_COMPLETE=>'已完成,待提交结算信息',
  82. self::STATUS_WAIT_SET=>'已提交结算信息,待后台结算',
  83. self::STATUS_SET=>'已结算',
  84. self::STATUS_SET_REJECT=>'已拒绝结算',
  85. self::STATUS_REFUND=>'退款处理中',
  86. self::STATUS_CANCEL=>'已取消',
  87. ];
  88. const REFUND_STATUS_DEFAULT=0;
  89. const REFUND_STATUS_PASS=1;
  90. const REFUND_STATUS_REJECT=2;
  91. public static $refundStatus=[
  92. self::REFUND_STATUS_DEFAULT=>'审核中',
  93. self::REFUND_STATUS_PASS=>'审核通过',
  94. self::REFUND_STATUS_REJECT=>'驳回',
  95. ];
  96. protected $append=[
  97. 'protect_valid',
  98. ];
  99. public function user(){
  100. return $this->belongsTo(User::class);
  101. }
  102. public function log(){
  103. return $this->hasMany(UserOrderLog::class,'order_id')/*->order('id','desc')*/;
  104. }
  105. public static function payed($params){
  106. $id=$params['id'];
  107. $order=self::find($id);
  108. $order['pay_time']=time();
  109. $order['status']=self::STATUS_WAIT_GET;
  110. $order->notifyAdmin();
  111. $order->save();
  112. }
  113. public function notifyAdmin(){
  114. try {
  115. $admins=config('site.basic_ordered_notify_admin');
  116. $from_city=$this['from_city'];
  117. $to_city=$this['to_city'];
  118. $send_date=$this['agree_date'];
  119. if($admins){
  120. $adminUsers=Admin::whereIn('id',$admins)->whereNotNull('mobile')->column('mobile');
  121. foreach ($adminUsers as $mobile){
  122. \app\common\library\Sms::send($mobile,null,'ordered',\app\common\library\Sms::orderParam($from_city,$to_city,$send_date));
  123. }
  124. }
  125. }catch (\Exception $e){
  126. user_log('sms_ordered',[
  127. 'file'=>$e->getFile(),
  128. 'msg'=>$e->getMessage(),
  129. 'trace'=>$e->getTrace(),
  130. ]);
  131. }
  132. }
  133. public function notifySender(){
  134. try {
  135. $users=$this->senderOrder()->group('user_id')->column('user_id');
  136. $from_city=$this['from_city'];
  137. $to_city=$this['to_city'];
  138. $send_date=$this['agree_date'];
  139. if($users) {
  140. $userMobile = User::sender()->whereIn('id',$users)->column('mobile');
  141. foreach ($userMobile as $mobile){
  142. \app\common\library\Sms::send($mobile,null,'dispatch',\app\common\library\Sms::orderDispatchParam($from_city,$to_city,$send_date));
  143. }
  144. }
  145. }catch (\Exception $e){
  146. user_log('sms_dispatch',[
  147. 'file'=>$e->getFile(),
  148. 'msg'=>$e->getMessage(),
  149. 'trace'=>$e->getTrace(),
  150. ]);
  151. }
  152. }
  153. public function setImagesAttr($v){
  154. if(!$v){
  155. $v=[];
  156. }
  157. return json_encode($v);
  158. }
  159. public function getImagesAttr($v){
  160. return array_filter(json_decode($v));
  161. }
  162. /**
  163. * 取消
  164. *user.type 1未支付2驳回3退款
  165. *user.reason
  166. */
  167. const CT_NOT_PAY=1;
  168. const CT_REJECT=2;
  169. const CT_REFUND=3;
  170. public function cancel($type=self::CT_NOT_PAY,$force=false){
  171. if($this['status']!==self::STATUS_WAIT_PAY && !$force){
  172. throw_user('非待付款订单无法取消');
  173. }
  174. $this['status']=self::STATUS_CANCEL;
  175. $this['cancel_at']=time();
  176. $this['cancel_type']=$type;
  177. $this->save();
  178. if($this['coupon_id']){
  179. UserCoupon::setUse($this['coupon_id'],0);
  180. }
  181. }
  182. public function pay(){
  183. if($this['status']!=self::STATUS_WAIT_PAY){
  184. throw_user('状态有误');
  185. }
  186. if($this['expired_at']<time()){
  187. $this->cancel();
  188. throw_user('该订单已超时');
  189. }
  190. if($this['pay_type']==1){
  191. $params='';
  192. $this->user->money(bcsub(0,$this['real_amount'],2),$this['user_id'],MoneyLog::TYPE_ORDER_PAY,"订单[{$this['no']}]付款");
  193. self::payed(['id'=>$this['id']]);
  194. }else {
  195. $params = Payment::pay($this->user, $this['real_amount'], self::class, 'payed', ['id' => $this['id']]);
  196. }
  197. return $params;
  198. }
  199. public function complete(){
  200. $this->refunding();
  201. if($this['status']!=self::STATUS_GIVED){
  202. throw_user('非已送达状态无法确认完成');
  203. }
  204. $this['status']=self::STATUS_COMPLETE;
  205. $this['completed_at']=time();
  206. if(!$this->save()){
  207. throw_user('保存状态失败');
  208. }
  209. }
  210. public function refund($reason,$images){
  211. if(!in_array($this['status'],[self::STATUS_WAIT_GET,self::STATUS_GETTING,self::STATUS_SENDING])){
  212. throw_user('当前无法申请退款');
  213. }
  214. $this['refund_reason']=$reason;
  215. $this['refund_apply_at']=time();
  216. //$this['refund_amount']=$this->refund_money;
  217. $this['refund_amount']=0;
  218. $this['refund_images']=$images;
  219. $this['refund_status']=self::REFUND_STATUS_DEFAULT;
  220. $this['status_bak']=$this['status'];
  221. $this['status']=self::STATUS_REFUND;
  222. if(!$this->save()){
  223. throw_user('保存失败');
  224. }
  225. }
  226. public function refundCancel(){
  227. if(!in_array($this['refund_status'],[self::REFUND_STATUS_DEFAULT])){
  228. throw_user('当前无法取消申请退款');
  229. }
  230. $this['refund_status']=-1;
  231. $this['status']=$this['status_bak'];
  232. if(!$this->save()){
  233. throw_user('保存失败');
  234. }
  235. }
  236. public function refunding(bool $throw=true){
  237. $is=in_array($this['refund_status'],[self::REFUND_STATUS_DEFAULT,self::REFUND_STATUS_PASS]);
  238. if($throw && $is){
  239. throw_user('当前正在申请退款或已退款,您无法操作');
  240. }
  241. return $is;
  242. }
  243. /**
  244. *处理退款申请
  245. * @status 1同意2拒绝
  246. */
  247. public function dealRefund($data){
  248. if($data['status']==1){
  249. $this['refund_status']=self::REFUND_STATUS_PASS;
  250. $this['refund_amount']=$data['amount'];
  251. $this['refund_at']=time();
  252. PayUser::setType($this['pay_type']);
  253. UserOrderStatic::addData($this,'refund',bcsub(0,$data['amount']));
  254. UserOrderStatic::addData($this,'refund_profit',bcsub($this['real_amount'],$data['amount']));
  255. PayUser::getClass()->payment($this['refund_amount'],$this['user'],MoneyLog::TYPE_REFUND,"订单[{$this['no']}]退款",['order_id'=>$this['id']]);
  256. $this->cancel(self::CT_REFUND,true);
  257. }elseif($data['status']==2){
  258. $this['refund_status']=self::REFUND_STATUS_REJECT;
  259. $this['status']=$this['status_bak'];
  260. }
  261. $this->save();
  262. }
  263. /**
  264. *处理结算申请
  265. * @status 1同意2拒绝
  266. */
  267. public function dealSettle($data){
  268. if($this['status']!=self::STATUS_WAIT_SET){
  269. throw_user(sprintf('非待结算订单无法结算[%s]',self::$statusList[$this['status']]));
  270. }
  271. if($data['status']==1){
  272. $this['status']=self::STATUS_SET;
  273. $settle_amount=0;
  274. foreach ($this->currentSenderOrder()->select() as $senderOrder){
  275. if(is_null($senderOrder['fee_total'])){
  276. throw_user('有配送员尚未申请结算,请稍后再试');
  277. }
  278. PayUser::getClass()->payment($senderOrder['fee_total'],$senderOrder['user'],MoneyLog::TYPE_SETTLE,"订单[{$this['no']}]结算",['order_id'=>$this['id']]);
  279. $settle_amount=bcadd($settle_amount,$senderOrder['fee_total']);
  280. }
  281. #加一个盈利记录
  282. User::money($this->profit,0,MoneyLog::TYPE_PROFIT,"订单[{$this['no']}]盈利",['order_id'=>$this['id']],false);
  283. UserOrderStatic::addData($this,'settle',bcsub(0,$settle_amount));
  284. UserOrderStatic::addData($this,'profit',$this->profit);
  285. $this['settled_at']=time();
  286. }elseif($data['status']==2){
  287. $this['status']=self::STATUS_COMPLETE;
  288. $this->currentSenderOrder()->update([
  289. 'fee_1'=>0,
  290. 'fee_2'=>0,
  291. 'fee_3'=>0,
  292. 'fee_4'=>0,
  293. 'fee_5'=>0,
  294. 'fee_6'=>0,
  295. 'fee_total'=>null,
  296. ]);
  297. }
  298. if(!$this->save()){
  299. throw_user('保存失败');
  300. }
  301. }
  302. public function getCalRefundAttr($v,$data){
  303. if($this['freight']==self::FREIGHT_AIR){
  304. $fee=bcdiv(config('site.order_refund_air'),100,4);
  305. }else{
  306. $fee=bcdiv(config('site.order_refund_land'),100,4);
  307. }
  308. $amount=bcmul($this['real_amount'],$fee);
  309. return $amount;
  310. }
  311. public function getRefundMoneyAttr($v,$data){
  312. return bcsub($this['real_amount'],$this->cal_refund);
  313. }
  314. public function getSendDetailAttr(){
  315. $log=$this->log()->user()->select();
  316. $res['addr']=$this['to_addr']['location'];
  317. $res['log']=$log;
  318. return $res;
  319. }
  320. public function remove(){
  321. if(!in_array($this['status'],[self::STATUS_CANCEL,self::STATUS_SET,self::STATUS_SET_REJECT])){
  322. throw_user('非已取消或完成订单无法删除');
  323. }
  324. if($this['user_deleted_at']){
  325. throw_user('该订单已删除');
  326. }
  327. /*$this['user_deleted_at']=time();
  328. $this->save();*/
  329. user_log('user_order_deleted',$this->toArray());
  330. $this->delete();
  331. }
  332. public function submit($data){
  333. $this['status']=self::STATUS_WAIT_PAY;
  334. #订单手动还是自动接
  335. $this['way']=(new OrderConfig())->get('order_way','auto');
  336. $res=$this->allowField(true)->save($data);
  337. if(!$res){
  338. throw_user('保存失败');
  339. }
  340. if($this['coupon_id']){
  341. UserCoupon::setUse($this['coupon_id']);
  342. }
  343. }
  344. /** 待接单 */
  345. public static function wait(User $user){
  346. $q=self::alias('user_order');
  347. $city=$user->area()->column('name');
  348. $sendType=$user->send()->column('type');
  349. if(!$sendType){
  350. $sendType=[0];
  351. }
  352. if(empty($city)){
  353. $q->where('user_order.id',0);
  354. return $q;
  355. }
  356. $q
  357. ->join('sender_order b','user_order.id=b.user_order_id and b.now=1','LEFT')
  358. ->whereIn('user_order.from_city|user_order.to_city',$city)
  359. ->whereIn('user_order.freight',$sendType)
  360. ->whereIn('user_order.way','auto')
  361. ->whereBetween('user_order.status',[self::STATUS_WAIT_GET,self::STATUS_GIVING])
  362. ->whereNotExists("select user_id from sender_order where user_order_id=user_order.id and user_id={$user['id']}")
  363. ->group('user_order.id')
  364. ->having('COUNT(b.id)<2')
  365. ->order('user_order.id','desc');
  366. return $q;
  367. }
  368. public function getProtectValidAttr($v,$d){
  369. if(!isset($this['status'])){
  370. return false;
  371. }
  372. return $this['status']>=self::STATUS_WAIT_GET && $this['status']<=self::STATUS_GIVED;
  373. }
  374. public function senderOrder(){
  375. return $this->hasMany(SenderOrder::class)->removeOption('soft_delete');
  376. }
  377. /** 当前的配送员订单 */
  378. public function currentSenderOrder($user_id=null){
  379. $q=$this->senderOrder()->where('now',1)->removeOption('soft_delete')->order('type');
  380. if($user_id){
  381. $q->where('user_id',$user_id);
  382. }
  383. return $q;
  384. }
  385. /** 是否能接单 */
  386. public function checkGet(User $user){
  387. if($this->currentSenderOrder()->count()>1){
  388. throw_user('订单已被接单,您暂时无法接单(1)');
  389. }
  390. $type=$user->getSendType($this);
  391. /*if($user->isSendGet($type) && $this['status'] != self::STATUS_WAIT_GET) {
  392. throw_user("订单已被接单,您暂时无法接单(2)");
  393. }*/
  394. return $type;
  395. }
  396. /** 接单 */
  397. public function accept(User $user){
  398. $type=$this->checkGet($user);
  399. if($this['status']==self::STATUS_WAIT_GET) {
  400. $this['status'] = self::STATUS_GETTING;
  401. }
  402. #取送都是自己,取送订单创建
  403. if($user->isSendAll($type)){
  404. $time=time();
  405. $get=$this->senderOrder()->save([
  406. 'user_id'=>$user['id'],
  407. 'status' =>$this['status'],
  408. 'type' =>1,
  409. 'now' =>1,
  410. 'get_at' =>$time,
  411. ]);
  412. if(!$get){
  413. throw_user('保存取宠订单失败');
  414. }
  415. $sender=$this->senderOrder()->save([
  416. 'user_id'=>$user['id'],
  417. 'status' =>$this['status'],
  418. 'type' =>2,
  419. 'now' =>1,
  420. 'get_at' =>$time,
  421. ]);
  422. if(!$sender){
  423. throw_user('保存送宠订单失败');
  424. }
  425. }else{
  426. $sender_order=$this->senderOrder()->save([
  427. 'user_id'=>$user['id'],
  428. 'status' =>$this['status'],
  429. 'type' =>$type,
  430. 'now' =>1,
  431. 'get_at' =>time(),
  432. ]);
  433. if(!$sender_order){
  434. throw_user('保存配送订单失败');
  435. }
  436. }
  437. if(!$this['get_at']){
  438. $this['get_at']=time();
  439. }
  440. $this->save();
  441. }
  442. public function manualGet(User $getUser,User $sendUser){
  443. //$getUser->checkOrderArea($this['from_city']);
  444. //$sendUser->checkOrderArea($this['to_city']);
  445. $this['status'] = self::STATUS_GETTING;
  446. #取送都是自己,取送订单创建
  447. $sender_order=$this->senderOrder()->save([
  448. 'user_id'=>$getUser['id'],
  449. 'status' =>$this['status'],
  450. 'type' =>1,
  451. 'now' =>1,
  452. 'get_at' =>time(),
  453. ]);
  454. if(!$sender_order){
  455. throw_user('保存配送订单1失败');
  456. }
  457. $sender_order=$this->senderOrder()->save([
  458. 'user_id'=>$sendUser['id'],
  459. 'status' =>$this['status'],
  460. 'type' =>2,
  461. 'now' =>1,
  462. 'get_at' =>time(),
  463. ]);
  464. if(!$sender_order){
  465. throw_user('保存配送订单2失败');
  466. }
  467. if(!$this['get_at']){
  468. $this['get_at']=time();
  469. }
  470. $this->save();
  471. $this->notifySender();
  472. }
  473. /** 更新状态 */
  474. public function updateStatus($status,User $user){
  475. if(in_array($status,[0,1])){
  476. return $status;
  477. }
  478. $this->refunding();
  479. $senderS=[
  480. self::STATUS_GETTING=>self::STATUS_SENDING,
  481. self::STATUS_SENDING=>self::STATUS_GIVING,
  482. ];
  483. $recS=[
  484. self::STATUS_GIVING=>self::STATUS_GIVED,
  485. ];
  486. $arr=$senderS+$recS;
  487. if($this['status']!=$status){
  488. throw_user('状态'.self::$statusList[$this['status']].'有误');
  489. }
  490. if(!$this->is_same_user) {
  491. $type = $user->getSendType($this);
  492. if ($user->isSendGet($type) && !isset($senderS[$status])) {
  493. throw_user('您是取宠人,无法更新状态' . self::$statusList[$status]);
  494. } elseif ($user->isSendSend($type) && !isset($recS[$status])) {
  495. throw_user('您是送宠人,无法更新状态' . self::$statusList[$status]);
  496. }
  497. }
  498. $this['status']=$arr[$status];
  499. if(!$this->save()){
  500. throw_user('更新失败');
  501. }
  502. return $this['status'];
  503. }
  504. /**
  505. * 进行中的订单
  506. * @param static|Query $query
  507. */
  508. public function scopeRunning(Query $query,$from=''){
  509. $status=[];
  510. if($from=='user'){
  511. $status=[
  512. self::STATUS_WAIT_GET,
  513. ];
  514. }
  515. $query
  516. ->show()
  517. ->whereIn('status',array_merge($status,[
  518. self::STATUS_GETTING,
  519. self::STATUS_SENDING,
  520. self::STATUS_GIVING,
  521. self::STATUS_GIVED,
  522. self::STATUS_REFUND,
  523. ]));
  524. }
  525. /** 计算收益 */
  526. public static function calcProfit($date=null){
  527. }
  528. /** 结算金额 */
  529. public function getSettleAmountAttr($v,$data){
  530. $sender_order=$this->currentSenderOrder()->select();
  531. $amount=0;
  532. foreach ($sender_order as $order){
  533. $amount=bcadd($amount,$order['fee_total']);
  534. }
  535. return $amount;
  536. }
  537. /** 配送员 */
  538. public function getSendersAttr($v,$data){
  539. $sender_order=$this->currentSenderOrder()->select();
  540. $new=[];
  541. #按0取1送的键值排序
  542. foreach ($sender_order as $item){
  543. $item->append(['user']);
  544. $new[$item['type']-1]=$item;
  545. }
  546. return array_column($new,null);
  547. }
  548. /** 结算总的金额 */
  549. public function getSettleInfoAttr($v,$data){
  550. $arr=$this->senders;
  551. $total=[
  552. 'fee_total'=>0
  553. ];
  554. foreach ($arr as $item){
  555. for ($i=1;$i<=6;$i++) {
  556. $total["fee_$i"] = bcadd($total["fee_$i"]??0,$item["fee_$i"]);
  557. }
  558. $total['fee_total']=bcadd($total['fee_total']??0,$item['fee_total']);
  559. }
  560. return $total;
  561. }
  562. /** 盈利总的金额 */
  563. public function getProfitAttr($v,$data){
  564. return bcsub($this['real_amount'],$this->settle_info['fee_total']);
  565. }
  566. /** 是否相同接单人 */
  567. public function getIsSameUserAttr(){
  568. $senders=$this->currentSenderOrder()->select();
  569. return $senders->count()==2 && count(array_unique($senders->column('user_id')))==1;
  570. }
  571. protected function getCreatedAtAttr($a){
  572. return date('Y-m-d H:i:s',$a)?:'';
  573. }
  574. protected function getPayTimeAttr($a){
  575. if(!$a){
  576. return '';
  577. }
  578. return date('Y-m-d H:i:s',$a);
  579. }
  580. protected function getCompletedAtAttr($a){
  581. if(!$a){
  582. return '';
  583. }
  584. return date('Y-m-d H:i:s',$a);
  585. }
  586. protected function getRefundApplyAtAttr($a){
  587. if(!$a){
  588. return '';
  589. }
  590. return date('Y-m-d H:i:s',$a);
  591. }
  592. protected function getRefundAtAttr($a){
  593. if(!$a){
  594. return '';
  595. }
  596. return date('Y-m-d H:i:s',$a);
  597. }
  598. /** init */
  599. protected static function init()
  600. {
  601. self::beforeInsert(function (self $order){
  602. $order['no']=order_no();
  603. $order['expired_at']=strtotime("+10minutes");
  604. });
  605. self::afterInsert(function (self $order){
  606. /* if(!$order->user->address()->exists($order['from_addr'])->find()) {
  607. $addr=$order['from_addr'];
  608. $addr['user_id']=$order->user_id;
  609. (new UserAddress())->allowField(true)->save($addr);
  610. }*/
  611. });
  612. self::afterUpdate(function (self $order){
  613. #更新配送订单状态
  614. $data=$order->getChangedData();
  615. if(isset($data['status'])){
  616. $order->senderOrder()->update(['status'=>$data['status']]);
  617. }
  618. #增加记录
  619. if(isset($data['status']) && $data['status']==self::STATUS_WAIT_GET){
  620. UserOrderStatic::addData($order,'pay',$order['real_amount']);
  621. }
  622. });
  623. self::afterDelete(function (self $order){
  624. user_log('user_order_deleted',$order->senderOrder()->select()->toArray());
  625. user_log('user_order_static',UserOrderStatic::where('no',$order['no'])->select()->toArray());
  626. $order->senderOrder()->delete();
  627. });
  628. }
  629. /** scope */
  630. /**
  631. * 待结算
  632. * @param Query $query
  633. */
  634. public function scopeSettle(Query $query){
  635. $query->where('status',self::STATUS_WAIT_SET);
  636. }
  637. /**
  638. * 待处理退款
  639. * @param Query $query
  640. */
  641. public function scopeWaitRefund(Query $query){
  642. $query->where('status',self::STATUS_REFUND);
  643. }
  644. /**
  645. * 售后
  646. * @param static $query
  647. */
  648. public function scopeService(Query $query){
  649. $query->show()->where('refund_status','>',-1);
  650. }
  651. /**
  652. * 总营业额
  653. * @param Query $query
  654. * @param null|array $date
  655. */
  656. public function scopeTurnover(Query $query,$date='today'){
  657. $query
  658. ->whereBetween(($table=$this->getTable()).'.status', [self::STATUS_WAIT_GET, self::STATUS_SET]);
  659. if($date=='today') {
  660. $query->whereTime($table.'.created_at', $date);
  661. }elseif(is_array($date)){
  662. list($type,$time)=$date;
  663. if($type=='day'){
  664. $query->whereBetween($table.'.created_at',[strtotime($time),strtotime("+1day",strtotime($time))]);
  665. }elseif ($type=='month'){
  666. if (substr_count($time,'-')==1){
  667. $time="$time-01 00:00:00";
  668. }
  669. $query->whereBetween($table.'.created_at',[strtotime($time),strtotime("+1month",strtotime($time))]);
  670. }elseif ($type=='year'){
  671. $query->whereBetween($table.'.created_at',[strtotime("$time-01-01"),strtotime(($time+1).'-01-01')]);
  672. }
  673. }
  674. }
  675. /**
  676. * 盈利统计
  677. * @param null $date
  678. * @param bool $detail
  679. * @return string
  680. * @throws \think\db\exception\DataNotFoundException
  681. * @throws \think\db\exception\ModelNotFoundException
  682. * @throws \think\exception\DbException
  683. */
  684. public function scopeProfit(Query $query,$date=null){
  685. $profit=$query
  686. ->join('sender_order b','user_order.id=b.user_order_id and b.now=1')
  687. ->where('user_order.status',self::STATUS_SET)
  688. ->show()
  689. ->group('user_order.id')
  690. ->fieldRaw("(user_order.real_amount-SUM(b.fee_total)) AS profit,SUM(b.fee_total) as fee_total");
  691. if(is_null($date)){
  692. $profit->whereTime('user_order.created_at','today');
  693. }else{
  694. list($type,$time)=$date;
  695. if($type=='day'){
  696. $profit->whereBetween('user_order.created_at',[$datTime=strtotime($time),$datTime+86400]);
  697. }elseif ($type=='month'){
  698. if (substr_count($time,'-')==1){
  699. $time="$time-01 00:00:00";
  700. }
  701. $profit->whereBetween('user_order.created_at',[strtotime($time),strtotime("+1month",strtotime($time))]);
  702. }elseif ($type=='year'){
  703. $profit->whereBetween('user_order.created_at',[strtotime("$time-01-01"),strtotime(($time+1).'-01-01')]);
  704. }
  705. }
  706. }
  707. /** 已付款,代接单 */
  708. public function scopeWaitGet(Query $query){
  709. $query
  710. ->show()
  711. ->where('status',self::STATUS_WAIT_GET);
  712. }
  713. /** 已完成 */
  714. public function scopeSuccess(Query $query){
  715. $query
  716. ->show()
  717. ->whereBetween('status',[
  718. self::STATUS_COMPLETE,
  719. self::STATUS_WAIT_SET,
  720. ]);
  721. }
  722. /** 已结算
  723. * @param Query|static $query
  724. */
  725. public function scopeSettled(Query $query){
  726. $query
  727. ->whereIn('status',[self::STATUS_SET,self::STATUS_SET_REJECT]);
  728. }
  729. /**
  730. * 已取消
  731. * @param Query|self $query
  732. */
  733. public function scopeCanceled(Query $query){
  734. $query->where('status',self::STATUS_CANCEL);
  735. }
  736. /** 已超时 */
  737. public function scopeExpired(Query $query,$limit=10){
  738. $query->where('expired_at','<',time())
  739. ->where('status',self::STATUS_WAIT_PAY)
  740. ->order('id','desc')
  741. ->limit($limit);
  742. }
  743. /**
  744. * 已完成
  745. * @param Query|self $query
  746. */
  747. public function scopeOver(Query $query){
  748. $query
  749. ->show()
  750. ->whereBetween('status',[self::STATUS_COMPLETE,self::STATUS_SET_REJECT]);
  751. }
  752. /** 展示的 */
  753. public function scopeShow(Query $query){
  754. $query->whereNull("{$this->getTable()}.user_deleted_at");
  755. }
  756. /** 后台使用已完成 */
  757. public function scopeCompleted(Query $query){
  758. $query->where('status',UserOrder::STATUS_COMPLETE)
  759. ->show()
  760. ->order('id','desc');
  761. }
  762. /** 后台使用已结算 */
  763. public function scopeCaled(Query $query){
  764. $query->where('status',UserOrder::STATUS_SET)
  765. ->show()
  766. ->order('id','desc');
  767. }
  768. /** 已付款的 */
  769. public function scopeHasPayed(Query $query){
  770. $query->whereNotNull('pay_time');
  771. }
  772. /** trashed */
  773. public static function withTrashed(){
  774. return new self;
  775. }
  776. }