截流自动化的商城平台
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.

OrderLogic.php 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | likeshop开源商城系统
  4. // +----------------------------------------------------------------------
  5. // | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
  6. // | gitee下载:https://gitee.com/likeshop_gitee
  7. // | github下载:https://github.com/likeshop-github
  8. // | 访问官网:https://www.likeshop.cn
  9. // | 访问社区:https://home.likeshop.cn
  10. // | 访问手册:http://doc.likeshop.cn
  11. // | 微信公众号:likeshop技术社区
  12. // | likeshop系列产品在gitee、github等公开渠道开源版本可免费商用,未经许可不能去除前后端官方版权标识
  13. // | likeshop系列产品收费版本务必购买商业授权,购买去版权授权后,方可去除前后端官方版权标识
  14. // | 禁止对系统程序代码以任何目的,任何形式的再发布
  15. // | likeshop团队版权所有并拥有最终解释权
  16. // +----------------------------------------------------------------------
  17. // | author: likeshop.cn.team
  18. // +----------------------------------------------------------------------
  19. namespace app\admin\logic\order;
  20. use app\common\basics\Logic;
  21. use app\common\enum\OrderEnum;
  22. use app\common\enum\ShopEnum;
  23. use app\common\model\order\Order;
  24. use app\common\enum\OrderLogEnum;
  25. use app\common\enum\PayEnum;
  26. use app\common\logic\OrderRefundLogic;
  27. use app\common\model\Express;
  28. use app\common\model\user\UserLevel;
  29. use app\common\server\ExportExcelServer;
  30. use app\common\server\UrlServer;
  31. use expressage\Kd100;
  32. use expressage\Kdniao;
  33. use app\common\server\ConfigServer;
  34. use think\facade\Db;
  35. use think\facade\Validate;
  36. /**
  37. * 订单管理-逻辑
  38. * Class GoodsLogic
  39. * @package app\shop\logic\goods
  40. */
  41. class OrderLogic extends Logic
  42. {
  43. /**
  44. * @notes 订单统计
  45. * @param array $get
  46. * @return array
  47. * @throws \think\db\exception\DataNotFoundException
  48. * @throws \think\db\exception\DbException
  49. * @throws \think\db\exception\ModelNotFoundException
  50. * @author suny
  51. * @date 2021/7/14 10:04 上午
  52. */
  53. public static function statistics($get = [], $is_export = false)
  54. {
  55. $order = new Order();
  56. $where = static::getWhere($get);
  57. $page = $get['page'] ?? 1;
  58. $limit = $get['limit'] ?? 10;
  59. if (Validate::must($get['page'] ?? '')) {
  60. $page = $get['page'];
  61. }
  62. if (Validate::must($get['limit'] ?? '')) {
  63. $limit = $get['limit'];
  64. }
  65. // 导出excel
  66. if (true === $is_export) {
  67. return self::export($where);
  68. }
  69. $field = 'o.*,s.name as shop_name,u.level';
  70. $count = $order
  71. ->alias('o')
  72. ->field($field)
  73. ->join('shop s', 's.id = o.shop_id')
  74. ->join('user u', 'u.id = o.user_id')
  75. ->join('order_goods g', 'g.order_id = o.id')
  76. ->with(['order_goods', 'user', 'shop'])
  77. ->where($where)
  78. ->group('o.id')
  79. ->count();
  80. $lists = $order
  81. ->alias('o')
  82. ->field($field)
  83. ->join('shop s', 's.id = o.shop_id')
  84. ->join('user u', 'u.id = o.user_id')
  85. ->join('order_goods g', 'g.order_id = o.id')
  86. ->with(['order_goods', 'user', 'shop'])
  87. ->where($where)
  88. ->append([
  89. 'delivery_address', 'pay_status_text', 'order_source_text',
  90. 'order_status_text', 'delivery_type_text','order_type_text',
  91. 'pay_way_text',
  92. ])
  93. ->page($page, $limit)
  94. ->order('o.id desc')
  95. ->group('o.id')
  96. ->select();
  97. $user_level = UserLevel::where(['del'=>0])->column('name','id');
  98. foreach ($lists as &$list) {
  99. $list['pay_time'] = $list['pay_time'] == '0' ? '未支付' : date('Y-m-d H:i:s', $list['pay_time']);
  100. $list['user']['avatar'] = UrlServer::getFileUrl($list['user']['avatar']);
  101. $list['shop']['logo'] = UrlServer::getFileUrl($list['shop']['logo']);
  102. foreach ($list['order_goods'] as $order_good) {
  103. $order_good['image'] = UrlServer::getFileUrl($order_good['image']);
  104. }
  105. // 会员等级
  106. $list['user_level'] = '暂无等级';
  107. if(isset($user_level[$list['level']])) {
  108. $list['user_level'] = $user_level[$list['level']];
  109. }
  110. //会员优惠
  111. $list['member_amount'] = $list['member_amount'] ?? 0.00;
  112. }
  113. return ['count' => $count, 'lists' => $lists];
  114. }
  115. static function getWhere($data)
  116. {
  117. $where = [];
  118. $where[] = [ 'o.del', '=', 0 ];
  119. $where[] = [ 'o.delete', '=', 0 ];
  120. //订单状态选项卡
  121. if (Validate::must($data['type'] ?? '')) {
  122. $where[] = [ 'order_status', '=', $data['type'] ];
  123. }
  124. //订单搜素
  125. if (! empty($data['search_key']) && ! empty($data['keyword'])) {
  126. $keyword = $data['keyword'];
  127. switch ($data['search_key']) {
  128. case 'order_sn':
  129. $where[] = ['o.order_sn', '=', $keyword];
  130. break;
  131. case 'user_sn':
  132. $where[] = ['u.sn', '=', $keyword];
  133. break;
  134. case 'shop_name':
  135. $where[] = ['s.name', 'like', '%' . $keyword . '%'];
  136. break;
  137. case 'goods_name':
  138. $where[] = ['g.goods_name', 'like', '%' . $keyword . '%'];
  139. break;
  140. case 'user_id':
  141. $where[] = ['o.user_id', '=', $keyword];
  142. break;
  143. case 'nickname':
  144. $where[] = ['u.nickname', 'like', '%' . $keyword . '%'];
  145. break;
  146. case 'user_mobile':
  147. $where[] = ['u.mobile', '=', $keyword];
  148. break;
  149. case 'consignee':
  150. $where[] = ['consignee', '=', $keyword];
  151. break;
  152. case 'consignee_mobile':
  153. $where[] = ['o.mobile', '=', $keyword];
  154. break;
  155. }
  156. }
  157. //商家名称
  158. if (Validate::must($data['shop_name'] ?? '')) {
  159. $where[] = [ 's.name', 'like', '%' . $data['shop_name'] . '%' ];
  160. }
  161. //商品名称
  162. if (Validate::must($data['goods_name'] ?? '')) {
  163. $where[] = ['g.goods_name', 'like', '%' . $data['goods_name'] . '%'];
  164. }
  165. //配送方式
  166. if (Validate::must($data['delivery_type'] ?? '')) {
  167. $where[] = ['o.delivery_type', '=', $data['delivery_type']];
  168. }
  169. //订单状态
  170. if (Validate::must($data['order_status'] ?? '')) {
  171. $where[] = [ 'o.order_status', '=', $data['order_status'] ];
  172. }
  173. //订单类型
  174. if (Validate::must($data['order_type'] ?? '')) {
  175. $where[] = [ 'o.order_type', '=', $data['order_type'] ];
  176. }
  177. //付款方式
  178. if (Validate::must($data['pay_way'] ?? '')) {
  179. $where[] = [ 'o.pay_way', '=', $data['pay_way'] ];
  180. }
  181. //订单来源
  182. if (Validate::must($data['order_source'] ?? '')) {
  183. $where[] = ['o.order_source', '=', $data['order_source']];
  184. }
  185. //下单时间
  186. if (Validate::must($data['start_time'] ?? '')) {
  187. $where[] = ['o.create_time', '>=', strtotime($data['start_time'])];
  188. }
  189. if (Validate::must($data['end_time'] ?? '')) {
  190. $where[] = ['o.create_time', '<=', strtotime($data['end_time'])];
  191. }
  192. return $where;
  193. }
  194. /**
  195. * @notes 订单详情
  196. * @param $id
  197. * @return array|\think\Model|null
  198. * @throws \think\db\exception\DataNotFoundException
  199. * @throws \think\db\exception\DbException
  200. * @throws \think\db\exception\ModelNotFoundException
  201. * @author suny
  202. * @date 2021/7/14 10:04 上午
  203. */
  204. public static function getDetail($id)
  205. {
  206. $order = new Order();
  207. $result = $order
  208. ->with(['user', 'order_goods', 'invoice'])
  209. ->where('id', $id)
  210. ->append(['delivery_address', 'pay_status_text', 'order_status_text', 'pay_way_text', 'order_type_text'])
  211. ->find();
  212. $result['pay_time'] = $result['pay_time'] == '0' ? '未支付' : date('Y-m-d H:i:s', $result['pay_time']);
  213. $result['user']['avatar'] = UrlServer::getFileUrl($result['user']['avatar']);
  214. foreach ($result['order_goods'] as &$order_goods) {
  215. $order_goods['goods_image'] = empty($order_goods['spec_image']) ?
  216. UrlServer::getFileUrl($order_goods['image']) : UrlServer::getFileUrl($order_goods['spec_image']);
  217. }
  218. // 自提提货时间
  219. if ($result['delivery_type'] == OrderEnum::DELIVERY_TYPE_SELF && $result['verification_status']) {
  220. $result['confirm_take_time'] = date('Y-m-d H:i:s', $result['confirm_take_time']);
  221. } else {
  222. $result['confirm_take_time'] = '';
  223. }
  224. //会员优惠
  225. $result['member_amount'] = $result['member_amount'] ?? 0.00;
  226. return $result;
  227. }
  228. /**
  229. * @notes 物流信息
  230. * @param $order_id
  231. * @return array|\PDOStatement|string|\think\Model|null
  232. * @throws \think\db\exception\DataNotFoundException
  233. * @throws \think\db\exception\ModelNotFoundException
  234. * @throws \think\exception\DbException
  235. * @author suny
  236. * @date 2021/7/14 10:05 上午
  237. */
  238. public static function shippingInfo($order_id)
  239. {
  240. $shipping = Db::name('delivery')->where('order_id', $order_id)->find();
  241. if ($shipping) {
  242. $shipping['create_time_text'] = date('Y-m-d H:i:s', $shipping['create_time']);
  243. }
  244. $shipping['traces'] = self::getShipping($order_id);
  245. return $shipping;
  246. }
  247. /**
  248. * @notes 物流轨迹
  249. * @param $order_id
  250. * @return bool|string[]
  251. * @throws \think\db\exception\DataNotFoundException
  252. * @throws \think\db\exception\DbException
  253. * @throws \think\db\exception\ModelNotFoundException
  254. * @author suny
  255. * @date 2021/7/14 10:05 上午
  256. */
  257. public static function getShipping($order_id)
  258. {
  259. $orderModel = new Order();
  260. $order = $orderModel->alias('o')
  261. ->field('invoice_no,shipping_name,shipping_id,o.shipping_status,o.mobile')
  262. ->join('delivery d', 'd.order_id = o.id')
  263. ->where(['o.id' => $order_id])
  264. ->find();
  265. $express = ConfigServer::get('express', 'way', '', '');
  266. $key = ConfigServer::get($express, 'appkey', '', '');
  267. $app = ConfigServer::get($express, 'appsecret', '', '');
  268. if (empty($express) || $order['shipping_status'] != 1 || empty($app) || empty($key)) {
  269. return $traces[] = ['暂无物流信息'];
  270. }
  271. //快递配置设置为快递鸟时
  272. if ($express === 'kdniao') {
  273. $expressage = (new Kdniao($app, $key, true));
  274. $shipping_field = 'codebird';
  275. } else {
  276. $expressage = (new Kd100($app, $key, true));
  277. $shipping_field = 'code100';
  278. }
  279. //快递编码
  280. $shipping_code = Db::name('express')
  281. ->where(['id' => $order['shipping_id']])
  282. ->value($shipping_field);
  283. //获取物流轨迹
  284. if (in_array(strtolower($shipping_code ), [ 'sf', 'shunfeng' ])) {
  285. if ($express === 'kdniao') {
  286. $expressage->logistics($shipping_code, $order['invoice_no'], substr($order['mobile'],-4));
  287. } else {
  288. $expressage->logistics($shipping_code, $order['invoice_no'], $order['mobile']);
  289. }
  290. }else {
  291. $expressage->logistics($shipping_code, $order['invoice_no']);
  292. }
  293. $traces = $expressage->logisticsFormat();
  294. if ($traces == false) {
  295. $traces[] = [$expressage->getError()];
  296. } else {
  297. foreach ($traces as &$item) {
  298. $item = array_values(array_unique($item));
  299. }
  300. }
  301. return $traces;
  302. }
  303. /**
  304. * @notes 获取物流
  305. * @return \think\Collection
  306. * @throws \think\db\exception\DataNotFoundException
  307. * @throws \think\db\exception\DbException
  308. * @throws \think\db\exception\ModelNotFoundException
  309. * @author suny
  310. * @date 2021/7/14 10:05 上午
  311. */
  312. public static function express()
  313. {
  314. return Express::where('del', 0)->select();
  315. }
  316. /**
  317. * @notes 取消订单(返回商品规格表库存)
  318. * @param $order_id
  319. * @param $admin_id
  320. * @return bool|string
  321. * @throws \think\db\exception\DataNotFoundException
  322. * @throws \think\db\exception\ModelNotFoundException
  323. * @throws \think\exception\DbException
  324. * @throws \think\exception\PDOException
  325. * @author suny
  326. * @date 2021/7/14 10:07 上午
  327. */
  328. public static function cancel($order_id, $admin_id)
  329. {
  330. Db::startTrans();
  331. try {
  332. $order = Order::where(['id' => $order_id], ['orderGoods'])->find();
  333. //取消订单
  334. OrderRefundLogic::cancelOrder($order_id, OrderLogEnum::TYPE_SHOP, $admin_id);
  335. //已支付的订单,取消,退款
  336. if ($order['pay_status'] == PayEnum::ISPAID) {
  337. //更新订单状态
  338. OrderRefundLogic::cancelOrderRefundUpdate($order);
  339. //订单退款
  340. OrderRefundLogic::refund($order, $order['order_amount'], $order['order_amount']);
  341. }
  342. Db::commit();
  343. return true;
  344. } catch (\Exception $e) {
  345. Db::rollback();
  346. //增加退款失败记录
  347. OrderRefundLogic::addErrorRefund($order, $e->getMessage());
  348. return $e->getMessage();
  349. }
  350. }
  351. /**
  352. * @notes 获取各列表数量
  353. * @return int
  354. * @author suny
  355. * @date 2021/7/14 10:08 上午
  356. */
  357. public static function getAll()
  358. {
  359. $data = input();
  360. unset($data['type']);
  361. return Order::alias('o')
  362. ->join('shop s', 's.id = o.shop_id')
  363. ->join('user u', 'u.id = o.user_id')
  364. ->join('order_goods g', 'g.order_id = o.id')
  365. ->with(['order_goods', 'user', 'shop'])
  366. ->where(static::getWhere($data))
  367. ->group('o.id')
  368. ->count();
  369. }
  370. /**
  371. * @notes 拼装统计信息
  372. * @param $order_status
  373. * @return array
  374. * @author suny
  375. * @date 2021/7/14 10:08 上午
  376. */
  377. public static function getStat($order_status)
  378. {
  379. $data = input();
  380. unset($data['type']);
  381. $result = [];
  382. foreach ($order_status as $status => $title) {
  383. $num = Order::alias('o')
  384. ->join('shop s', 's.id = o.shop_id')
  385. ->join('user u', 'u.id = o.user_id')
  386. ->join('order_goods g', 'g.order_id = o.id')
  387. ->with(['order_goods', 'user', 'shop'])
  388. ->where(static::getWhere($data))
  389. ->where('o.order_status', $status)
  390. ->group('o.id')
  391. ->count();
  392. $result[] = [
  393. 'title' => $title,
  394. 'status' => $status,
  395. 'count' => $num,
  396. ];
  397. }
  398. return $result;
  399. }
  400. /**
  401. * @notes 导出Excel
  402. * @param array $condition
  403. * @return array|false
  404. * @author 段誉
  405. * @date 2022/4/24 10:10
  406. */
  407. public static function export($condition = [])
  408. {
  409. try {
  410. $field = 'o.*,order_status as order_status_text,pay_way as pay_way_text,
  411. o.delivery_type as delivery_type_text,order_type as order_type_text,
  412. u.nickname,s.name as shop_name,s.type as shop_type';
  413. $lists = Order::alias('o')
  414. ->join('shop s', 's.id = o.shop_id')
  415. ->join('user u', 'u.id = o.user_id')
  416. ->join('order_goods g', 'g.order_id = o.id')
  417. ->with(['order_goods'])
  418. ->field($field)
  419. ->where($condition)
  420. ->append(['delivery_address', 'pay_status_text', 'order_source_text'])
  421. ->order('o.id desc')
  422. ->group('o.id')
  423. ->select()
  424. ->toArray();
  425. foreach ($lists as &$item) {
  426. $orderGoodsList = [];
  427. $goodsItemList = [];
  428. $goodsPriceList = [];
  429. $goodsNumList = [];
  430. foreach ($item['order_goods'] as $good) {
  431. $orderGoodsList[] = $good['goods_name'];
  432. $goodsItemList[] = $good['spec_value'];
  433. $goodsPriceList[] = $good['goods_price'];
  434. $goodsNumList[] = $good['goods_num'];
  435. }
  436. $item['order_goods_list'] = implode(';', $orderGoodsList);
  437. $item['goods_item_list'] = implode(';', $goodsItemList);
  438. $item['goods_price_list'] = implode(';', $goodsPriceList);
  439. $item['goods_num_list'] = implode(';', $goodsNumList);
  440. $item['shop_type'] = ShopEnum::getShopTypeDesc($item['shop_type']);
  441. }
  442. $excelFields = [
  443. 'shop_name' => '商家名称',
  444. 'shop_type' => '商家类型',
  445. 'order_sn' => '订单编号',
  446. 'order_type_text' => '订单类型',
  447. 'nickname' => '用户名称',
  448. 'order_goods_list' => '商品信息',
  449. 'goods_item_list' => '规格',
  450. 'goods_price_list' => '商品价格',
  451. 'goods_num_list' => '商品数量',
  452. 'order_amount' => '实付金额',
  453. 'consignee' => '收货人',
  454. 'mobile' => '收货人手机',
  455. 'delivery_address' => '收货地址',
  456. 'pay_status_text' => '支付状态',
  457. 'order_status_text' => '订单状态',
  458. 'create_time' => '下单时间',
  459. ];
  460. $export = new ExportExcelServer();
  461. $export->setFileName('订单');
  462. $result = $export->createExcel($excelFields, $lists);
  463. return ['url' => $result];
  464. } catch (\Exception $e) {
  465. self::$error = $e->getMessage();
  466. return false;
  467. }
  468. }
  469. }