Ingen beskrivning
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

OrderPreHandle.php 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海南赞赞网络科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 陈风任 <491085389@qq.com>
  11. * Date: 2021-04-27
  12. */
  13. namespace app\common\model;
  14. use think\Db;
  15. use think\Model;
  16. /**
  17. * 订单预处理模型
  18. */
  19. class OrderPreHandle extends Model
  20. {
  21. //初始化
  22. protected function initialize()
  23. {
  24. // 需要调用`Model`的`initialize`方法
  25. parent::initialize();
  26. // 接收参数
  27. $this->users_id = 0;
  28. $this->usersConfig = [];
  29. }
  30. // 订单预处理 (自动关闭未付款订单 发货后自动确认收货 收货后超过维权时间则关闭维权入口)
  31. public function eyou_shopOrderPreHandle($users_id = 0, $usersConfig = [], $action = 'users')
  32. {
  33. // 参数处理
  34. $this->users_id = !empty($users_id) ? intval($users_id) : 0;
  35. $this->usersConfig = !empty($usersConfig) ? $usersConfig : getUsersConfigData('all');
  36. // 超过 未付款自动关闭时间 则修改为已订单过期,仅针对待付款订单
  37. if (!empty($this->usersConfig['order_unpay_close_time'])) $this->eyou_paymentOrderHandle();
  38. // 超过 发货后自动收货时间 则修改自动确认收货,仅针对待收货订单
  39. if (!empty($this->usersConfig['order_auto_receipt_time'])) $this->eyou_receivedOrderHandle();
  40. // 超过 收货后可维权时间 则修改自动更新为不允许申请维权,仅针对已收货订单
  41. if (!empty($this->usersConfig['order_right_protect_time'])) $this->eyou_receiptOrderHandle();
  42. // 查询 (需要赠送积分 && 不可维权) 订单,执行赠送积分
  43. $this->eyou_consumObtainScoresHandle();
  44. // 查询 分销商的分销订单执行分销订单结算分佣处理
  45. if ('users' == $action) $this->eyou_dealerOrderSettlementHandle();
  46. }
  47. // 超过 未付款自动关闭时间 则修改为已订单过期,仅针对待付款订单
  48. private function eyou_paymentOrderHandle()
  49. {
  50. // 计算订单过期时间
  51. $orderUnpayCloseTime = intval($this->usersConfig['order_unpay_close_time']) * 60;
  52. // 查询过期的订单
  53. $time = getTime() - intval($orderUnpayCloseTime);
  54. $where = [
  55. 'order_status' => 0,
  56. 'add_time' => ['<', $time],
  57. ];
  58. //秒杀订单走秒杀独立的关闭逻辑
  59. if (is_dir('./weapp/Seckill/')) {
  60. $where['is_seckill_order'] = 0;
  61. }
  62. // 查询条件-会员ID处理
  63. if (!empty($this->users_id)) $where['users_id'] = $this->users_id;
  64. $shopOrder = Db::name('shop_order')->where($where)->select();
  65. if (!empty($shopOrder)) {
  66. $where = [
  67. 'order_id' => ['IN', get_arr_column($shopOrder, 'order_id')]
  68. ];
  69. // 恢复优惠券
  70. $this->restoreCoupon($where);
  71. // 商品库存恢复
  72. // $orderStock = $shopOrder;
  73. // foreach ($orderStock as $key => $value) {
  74. // // 如果订单是付款减库存则去除,订单过期不恢复库存
  75. // if (isset($value['order_stock_type']) && 1 === intval($value['order_stock_type'])) unset($orderStock[$key]);
  76. // }
  77. $this->restoreGoodsStock($shopOrder);
  78. // 删除未付款订单及关联数据
  79. Db::name('shop_order')->where($where)->delete(true);
  80. Db::name('shop_order_log')->where($where)->delete(true);
  81. Db::name('shop_order_details')->where($where)->delete(true);
  82. // 更新订单为已过期
  83. // $update = [
  84. // 'order_status' => 4,
  85. // 'update_time' => getTime(),
  86. // ];
  87. // Db::name('shop_order')->where($where)->update($update);
  88. // 添加订单操作记录
  89. // $actionNote = '订单未在' . $this->usersConfig['order_unpay_close_time'] . '分钟内完成支付,系统自动关闭!';
  90. // AddOrderAction($shopOrder, 0, 0, 4, 0, 0, '订单过期', $actionNote);
  91. }
  92. //秒杀订单走秒杀独立的关闭逻辑
  93. if (is_dir('./weapp/Seckill/')) {
  94. $seckill_config = getUsersConfigData('seckill');
  95. if (!empty($seckill_config['seckill_close_order_type']) && !empty($seckill_config['seckill_close_order_time'])){
  96. $time = intval($seckill_config['seckill_close_order_time']) * 60;
  97. $where = [
  98. 'order_status' => 0,
  99. 'add_time' => ['<', $time],
  100. 'is_seckill_order' => ['>', 0],
  101. ];
  102. $shopOrder = Db::name('shop_order')->field('order_id,users_id')->where($where)->select();
  103. if (!empty($shopOrder)) {
  104. $del_where = [
  105. 'order_id' => ['IN', get_arr_column($shopOrder, 'order_id')]
  106. ];
  107. // 调用秒杀逻辑层方法
  108. $weappSeckillLogic = new \weapp\Seckill\logic\SeckillLogic;
  109. foreach ($shopOrder as $k => $v){
  110. $weappSeckillLogic->cancelOrderHandle($v['order_id'], $v['users_id']);
  111. }
  112. // 删除未付款订单及关联数据
  113. Db::name('shop_order')->where($del_where)->delete(true);
  114. Db::name('shop_order_log')->where($del_where)->delete(true);
  115. Db::name('shop_order_details')->where($del_where)->delete(true);
  116. }
  117. }
  118. }
  119. }
  120. // 恢复优惠券
  121. private function restoreCoupon($where = [])
  122. {
  123. $useID = Db::name('shop_order')->where($where)->column('use_id');
  124. if (!empty($useID)) {
  125. $times = getTime();
  126. $where = [
  127. 'use_id' => ['IN', $useID]
  128. ];
  129. $couponUse = Db::name('shop_coupon_use')->where($where)->select();
  130. if (!empty($couponUse)) {
  131. foreach ($couponUse as $key => $value) {
  132. $where = [
  133. 'use_id' => $value['use_id'],
  134. 'users_id' => $value['users_id'],
  135. ];
  136. $update = [
  137. 'use_time' => 0,
  138. 'use_status' => 2,
  139. 'update_time' => $times
  140. ];
  141. if ($value['start_time'] <= $times && $value['end_time'] >= $times) $update['use_status'] = 0;
  142. Db::name('shop_coupon_use')->where($where)->update($update);
  143. }
  144. }
  145. }
  146. }
  147. // 恢复商品库存
  148. public function restoreGoodsStock($order_id)
  149. {
  150. // 查询订单商品
  151. $where = [];
  152. // 查询条件-会员ID处理
  153. if (!empty($this->users_id)) $where['a.users_id'] = $this->users_id;
  154. // 查询条件-订单ID处理
  155. if (is_array($order_id)) {
  156. $where['a.order_id'] = ['IN', get_arr_column($order_id, 'order_id')];
  157. } else {
  158. $where['a.order_id'] = $order_id;
  159. }
  160. $field = 'a.product_id, a.num as product_num, a.data as product_data, b.aid, b.value_id, b.spec_value_id';
  161. $orderDetails = Db::name('shop_order_details')
  162. ->alias('a')
  163. ->where($where)
  164. ->field($field)
  165. ->join('__PRODUCT_SPEC_VALUE__ b', 'a.product_id = b.aid', 'LEFT')
  166. ->select();
  167. if (!empty($orderDetails)) {
  168. // 循环组装数据
  169. $arcData = $specData = $pointsGoods = [];
  170. foreach ($orderDetails as $key => $value) {
  171. $productData = !empty($value['product_data']) ? unserialize($value['product_data']) : [];
  172. if (!empty($productData['pointsGoodsBuyField'])) {
  173. !in_array($productData['pointsGoodsBuyField'], $pointsGoods) && array_push($pointsGoods, $productData['pointsGoodsBuyField']);
  174. } else {
  175. $spec_value_id = !empty($productData['spec_value_id']) ? $productData['spec_value_id'] : 0;
  176. if (!empty($value['value_id']) && !empty($value['spec_value_id'])) {
  177. $where = [
  178. 'aid' => $value['product_id'],
  179. 'spec_value_id' => $spec_value_id,
  180. ];
  181. $value_id = Db::name('product_spec_value')->where($where)->getField('value_id');
  182. if (intval($value['value_id']) === intval($value_id) && intval($value['spec_value_id']) === intval($spec_value_id)) {
  183. // 有规格
  184. $specData[] = [
  185. 'value_id' => $value['value_id'],
  186. 'spec_stock' => Db::raw('spec_stock+' . ($value['product_num'])),
  187. 'spec_sales_num' => Db::raw('spec_sales_num-' . ($value['product_num'])),
  188. ];
  189. // 无规格
  190. $arcData[] = [
  191. 'aid' => $value['product_id'],
  192. 'stock_count' => Db::raw('stock_count+' . ($value['product_num'])),
  193. 'sales_num' => Db::raw('sales_num-' . ($value['product_num']))
  194. ];
  195. }
  196. } else {
  197. // 无规格
  198. $arcData[] = [
  199. 'aid' => $value['product_id'],
  200. 'stock_count' => Db::raw('stock_count+' . ($value['product_num'])),
  201. 'sales_num' => Db::raw('sales_num-' . ($value['product_num']))
  202. ];
  203. }
  204. }
  205. }
  206. // 更新规格库存销量
  207. if (!empty($specData)) {
  208. $productSpecValueModel = new \app\user\model\ProductSpecValue();
  209. $productSpecValueModel->saveAll($specData);
  210. Db::name('product_spec_value')->where(['spec_sales_num'=>['lt',0]])->update(['spec_sales_num'=>0, 'update_time'=>getTime()]);
  211. }
  212. // 更新商品库存销量
  213. if (!empty($arcData)) {
  214. $archivesModel = new \app\user\model\Archives();
  215. $archivesModel->saveAll($arcData);
  216. Db::name('archives')->where(['sales_num'=>['lt',0]])->update(['sales_num'=>0, 'update_time'=>getTime()]);
  217. }
  218. // 积分商品库存处理
  219. if (!empty($pointsGoods)) {
  220. $weappInfo = model('ShopPublicHandle')->getWeappPointsShop();
  221. if (!empty($weappInfo)) {
  222. $pointsGoodsModel = new \app\plugins\model\PointsGoods();
  223. $pointsGoodsModel->updatePointsGoodsStock($pointsGoods, 'increase');
  224. }
  225. }
  226. }
  227. }
  228. // 超过 发货后自动收货时间 则修改自动确认收货,仅针对待收货订单
  229. private function eyou_receivedOrderHandle()
  230. {
  231. // 计算订单自动收货时间
  232. $orderAutoReceiptTime = intval($this->usersConfig['order_auto_receipt_time']) * 86400;
  233. // 查询待收货订单
  234. $time = getTime() - intval($orderAutoReceiptTime);
  235. $where = [
  236. // 'prom_type' => 0,
  237. 'order_status' => 2,
  238. 'express_time' => ['<', $time],
  239. ];
  240. // 查询条件-会员ID处理
  241. if (!empty($this->users_id)) $where['users_id'] = $this->users_id;
  242. $order_ids = Db::name('shop_order')->field('order_id')->where($where)->select();
  243. if (!empty($order_ids)) {
  244. // 更新订单为已收货
  245. $update = [
  246. 'order_status' => 3,
  247. 'update_time' => getTime(),
  248. 'confirm_time' => getTime(),
  249. ];
  250. Db::name('shop_order')->where($where)->update($update);
  251. // 添加订单操作记录
  252. $actionNote = '订单已超过' . $this->usersConfig['order_auto_receipt_time'] . '天,用户未确认收货,系统自动收货!';
  253. AddOrderAction($order_ids, 0, 0, 3, 0, 0, '自动收货', $actionNote);
  254. }
  255. }
  256. // 超过 收货后可维权时间 则修改自动更新为不允许申请维权,仅针对已收货订单
  257. private function eyou_receiptOrderHandle()
  258. {
  259. // 计算订单可维权时间
  260. $orderRightProtectTime = intval($this->usersConfig['order_right_protect_time']) * 86400;
  261. // 查询待收货订单
  262. $time = getTime() - intval($orderRightProtectTime);
  263. $where = [
  264. 'order_status' => 3,
  265. 'allow_service' => 0,
  266. 'confirm_time' => ['<', $time],
  267. ];
  268. // 查询条件-会员ID处理
  269. if (!empty($this->users_id)) $where['users_id'] = $this->users_id;
  270. $order_ids = Db::name('shop_order')->field('order_id')->where($where)->select();
  271. if (!empty($order_ids)) {
  272. // 更新订单为不允许申请维权
  273. $update = [
  274. 'allow_service' => 1,
  275. 'update_time' => getTime(),
  276. ];
  277. Db::name('shop_order')->where($where)->update($update);
  278. // 添加订单操作记录
  279. $actionNote = '订单已超过' . $this->usersConfig['order_right_protect_time'] . '天,不再允许申请售后维权!';
  280. AddOrderAction($order_ids, 0, 0, 3, 0, 0, '关闭维权', $actionNote);
  281. }
  282. }
  283. // 查询 (需要赠送积分 && 不可维权) 订单,执行赠送积分
  284. private function eyou_consumObtainScoresHandle()
  285. {
  286. // 查询订单
  287. $where = [
  288. 'order_status' => 3,
  289. 'allow_service' => 1,
  290. 'is_obtain_scores' => 0,
  291. 'obtain_scores' => ['>', 0],
  292. ];
  293. // 查询条件-会员ID处理
  294. if (!empty($this->users_id)) $where['users_id'] = $this->users_id;
  295. // 查询订单id数组用于添加订单操作记录
  296. $shopOrder = Db::name('shop_order')->where($where)->select();
  297. if (!empty($shopOrder)) {
  298. foreach ($shopOrder as $key => $value) {
  299. if (!empty($value['users_id']) && !empty($value['obtain_scores'])) {
  300. // 赠送会员积分
  301. $insert = [
  302. 'type' => 8,
  303. 'users_id' => $value['users_id'],
  304. 'score' => $value['obtain_scores'],
  305. 'info' => '商城消费赠送',
  306. 'remark' => '商城消费赠送',
  307. ];
  308. addConsumObtainScores($insert, 2, true);
  309. // 添加订单操作记录
  310. if (!empty($value['order_id'])) {
  311. $actionNote = '订单已完成,赠送会员' . $value['obtain_scores'] . $this->usersConfig['score_name'];
  312. AddOrderAction($value['order_id'], 0, 0, 3, 1, 1, '消费赠送', $actionNote);
  313. }
  314. }
  315. }
  316. // 批量修改订单状态
  317. $update = [
  318. 'is_obtain_scores' => 1,
  319. 'update_time' => getTime(),
  320. ];
  321. Db::name('shop_order')->where($where)->update($update);
  322. }
  323. }
  324. // 查询 分销商的分销订单执行分销订单结算分佣处理
  325. private function eyou_dealerOrderSettlementHandle()
  326. {
  327. // 如果安装了分销插件则执行
  328. if (is_dir('./weapp/DealerPlugin/')) {
  329. // 开启分销插件则执行
  330. $data = model('Weapp')->getWeappList('DealerPlugin');
  331. if (!empty($data['status']) && 1 == $data['status']) {
  332. // 调用分销逻辑层方法
  333. $dealerCommonLogic = new \weapp\DealerPlugin\logic\DealerCommonLogic;
  334. $orderData = $dealerCommonLogic->dealerOrderSettlementHandle($this->users_id, $this->usersConfig);
  335. }
  336. }
  337. }
  338. }