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

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195
  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\shop\logic\order;
  20. use app\admin\logic\distribution\DistributionLevelLogic;
  21. use app\api\logic\PayLogic;
  22. use app\common\basics\Logic;
  23. use app\common\enum\NoticeEnum;
  24. use app\common\enum\OrderEnum;
  25. use app\common\enum\OrderLogEnum;
  26. use app\common\enum\PayEnum;
  27. use app\common\enum\TeamEnum;
  28. use app\common\enum\VerificationEnum;
  29. use app\common\logic\GoodsVirtualLogic;
  30. use app\common\logic\OrderLogLogic;
  31. use app\common\logic\OrderRefundLogic;
  32. use app\common\logic\PayNotifyLogic;
  33. use app\common\model\Delivery;
  34. use app\common\model\Express;
  35. use app\common\model\goods\Goods;
  36. use app\common\model\order\Order;
  37. use app\common\model\DevRegion;
  38. use app\common\model\order\OrderGoods;
  39. use app\common\model\printer\Printer;
  40. use app\common\model\printer\PrinterConfig;
  41. use app\common\model\team\Team;
  42. use app\common\model\team\TeamFound;
  43. use app\common\model\team\TeamJoin;
  44. use app\common\model\user\User;
  45. use app\common\model\user\UserLevel;
  46. use app\common\model\order\Verification;
  47. use app\common\server\ConfigServer;
  48. use app\common\server\ExportExcelServer;
  49. use app\common\server\UrlServer;
  50. use app\common\server\YlyPrinter;
  51. use app\shop\logic\printer\PrinterLogic;
  52. use think\facade\Cache;
  53. use expressage\Kd100;
  54. use expressage\Kdniao;
  55. use think\facade\Db;
  56. use think\Exception;
  57. use think\facade\Validate;
  58. use think\Model;
  59. /**
  60. * 订单管理-逻辑
  61. * Class GoodsLogic
  62. * @package app\shop\logic\goods
  63. */
  64. class OrderLogic extends Logic
  65. {
  66. /**
  67. * @notes 订单统计
  68. * @param array $get
  69. * @param int $shop_id
  70. * @return array
  71. * @throws \think\db\exception\DataNotFoundException
  72. * @throws \think\db\exception\DbException
  73. * @throws \think\db\exception\ModelNotFoundException
  74. * @author suny
  75. * @date 2021/7/14 10:23 上午
  76. */
  77. public static function statistics(array $get = [], int $shop_id = 0, $is_export = false)
  78. {
  79. $order = new Order();
  80. $where = static::getWhere($get);
  81. $page = $get['page'] ?? 1;
  82. $limit = $get['limit'] ?? 10;
  83. $where[] = ['o.delete', '=', 0];
  84. $where[] = ['o.del', '=', 0];
  85. $where[] = ['o.shop_id', '=', $shop_id];
  86. if (Validate::must($get['page'] ?? '')) {
  87. $page = $get['page'];
  88. }
  89. if (Validate::must($get['limit'] ?? '')) {
  90. $limit = $get['limit'];
  91. }
  92. // 导出excel
  93. if (true === $is_export) {
  94. return self::export($where);
  95. }
  96. $field = 'o.*,s.name as shop_name,u.level';
  97. $count = $order
  98. ->alias('o')
  99. ->Join('shop s', 's.id = o.shop_id')
  100. ->Join('user u', 'u.id = o.user_id')
  101. ->Join('order_goods g', 'g.order_id = o.id')
  102. ->where($where)
  103. ->group('o.id')
  104. ->count();
  105. $lists = $order
  106. ->alias('o')
  107. ->field($field)
  108. ->Join('shop s', 's.id = o.shop_id')
  109. ->Join('user u', 'u.id = o.user_id')
  110. ->Join('order_goods g', 'g.order_id = o.id')
  111. ->with(['order_goods', 'user'])
  112. ->where($where)
  113. ->append([
  114. 'delivery_address', 'pay_status_text', 'order_source_text',
  115. 'order_status_text', 'delivery_type_text','order_type_text',
  116. 'pay_way_text', 'is_team_success', 'to_ship_btn', 'to_verification_btn', 'edit_address_btn',
  117. ])
  118. ->page($page, $limit)
  119. ->order('o.id desc')
  120. ->group('o.id')
  121. ->select()->toArray();
  122. $user_level = UserLevel::where(['del'=>0])->column('name','id');
  123. foreach ($lists as &$list) {
  124. $list['pay_time'] = $list['pay_time'] == '0' ? '未支付' : date('Y-m-d H:i:s', $list['pay_time']);
  125. $list['user']['avatar'] = UrlServer::getFileUrl($list['user']['avatar']);
  126. foreach ($list['order_goods'] as $order_good) {
  127. $order_good['image'] = UrlServer::getFileUrl($order_good['image']);
  128. }
  129. // 订单核销按钮 & 发货按钮 & 修改地址
  130. $list['verification_btn'] = $list['to_verification_btn'];
  131. $list['send_btn'] = $list['to_ship_btn'];
  132. $list['change_address_btn'] = $list['edit_address_btn'];
  133. // 会员等级
  134. $list['user_level'] = '暂无等级';
  135. if(isset($user_level[$list['level']])) {
  136. $list['user_level'] = $user_level[$list['level']];
  137. }
  138. //会员优惠
  139. $list['member_amount'] = $list['member_amount'] ?? 0.00;
  140. }
  141. return ['count' => $count, 'lists' => $lists];
  142. }
  143. static function getWhere($data)
  144. {
  145. $where = [];
  146. $where[] = [ 'o.del', '=', 0 ];
  147. $where[] = [ 'o.delete', '=', 0 ];
  148. //订单状态选项卡
  149. if (Validate::must($data['type'] ?? '')) {
  150. $where[] = [ 'order_status', '=', $data['type'] ];
  151. }
  152. //订单搜素
  153. if (! empty($data['search_key']) && ! empty($data['keyword'])) {
  154. $keyword = $data['keyword'];
  155. switch ($data['search_key']) {
  156. case 'order_sn':
  157. $where[] = ['o.order_sn', '=', $keyword];
  158. break;
  159. case 'user_sn':
  160. $where[] = ['u.sn', '=', $keyword];
  161. break;
  162. case 'shop_name':
  163. $where[] = ['s.name', 'like', '%' . $keyword . '%'];
  164. break;
  165. case 'goods_name':
  166. $where[] = ['g.goods_name', 'like', '%' . $keyword . '%'];
  167. break;
  168. case 'user_id':
  169. $where[] = ['o.user_id', '=', $keyword];
  170. break;
  171. case 'nickname':
  172. $where[] = ['u.nickname', 'like', '%' . $keyword . '%'];
  173. break;
  174. case 'user_mobile':
  175. $where[] = ['u.mobile', '=', $keyword];
  176. break;
  177. case 'consignee':
  178. $where[] = ['consignee', '=', $keyword];
  179. break;
  180. case 'consignee_mobile':
  181. $where[] = ['o.mobile', '=', $keyword];
  182. break;
  183. }
  184. }
  185. //商品名称
  186. if (Validate::must($data['goods_name'] ?? '')) {
  187. $where[] = ['g.goods_name', 'like', '%' . $data['goods_name'] . '%'];
  188. }
  189. //配送方式
  190. if (Validate::must($data['delivery_type'] ?? '')) {
  191. $where[] = ['o.delivery_type', '=', $data['delivery_type']];
  192. }
  193. //订单状态
  194. if (Validate::must($data['order_status'] ?? '')) {
  195. $where[] = [ 'o.order_status', '=', $data['order_status'] ];
  196. }
  197. //订单类型
  198. if (Validate::must($data['order_type'] ?? '')) {
  199. $where[] = [ 'o.order_type', '=', $data['order_type'] ];
  200. }
  201. //付款方式
  202. if (Validate::must($data['pay_way'] ?? '')) {
  203. $where[] = [ 'o.pay_way', '=', $data['pay_way'] ];
  204. }
  205. //订单来源
  206. if (Validate::must($data['order_source'] ?? '')) {
  207. $where[] = ['o.order_source', '=', $data['order_source']];
  208. }
  209. //下单时间
  210. if (Validate::must($data['start_time'] ?? '')) {
  211. $where[] = ['o.create_time', '>=', strtotime($data['start_time'])];
  212. }
  213. if (Validate::must($data['end_time'] ?? '')) {
  214. $where[] = ['o.create_time', '<=', strtotime($data['end_time'])];
  215. }
  216. return $where;
  217. }
  218. public static function totalCount($shopId)
  219. {
  220. $all = Order::where(['shop_id' => $shopId, 'del' => 0, 'delete' => 0])->count();
  221. $waitPay = Order::where(['shop_id' => $shopId, 'order_status' => 0 , 'del' => 0, 'delete' => 0])->count();
  222. $waitDelivery = Order::where(['shop_id' => $shopId, 'order_status' => 1 , 'del' => 0, 'delete' => 0])->count();
  223. $waitReceipt = Order::where(['shop_id' => $shopId, 'order_status' => 2 , 'del' => 0, 'delete' => 0])->count();
  224. $waitComplete = Order::where(['shop_id' => $shopId, 'order_status' => 3 , 'del' => 0, 'delete' => 0])->count();
  225. $waitClose = Order::where(['shop_id' => $shopId, 'order_status' => 4 , 'del' => 0, 'delete' => 0])->count();
  226. return [
  227. 'all' => $all,
  228. 'wait_pay' => $waitPay,
  229. 'wait_delivery' => $waitDelivery,
  230. 'wait_receipt' => $waitReceipt,
  231. 'wait_complete' => $waitComplete,
  232. 'wait_close' => $waitClose,
  233. ];
  234. }
  235. /**
  236. * @notes 订单详情
  237. * @param $id
  238. * @return array|Model|null
  239. * @throws \think\db\exception\DataNotFoundException
  240. * @throws \think\db\exception\DbException
  241. * @throws \think\db\exception\ModelNotFoundException
  242. * @author suny
  243. * @date 2021/7/14 10:23 上午
  244. */
  245. public static function getDetail($id)
  246. {
  247. $order = new Order();
  248. $result = $order
  249. ->with(['user', 'order_goods', 'invoice'])
  250. ->where('id', $id)
  251. ->append(['delivery_address', 'pay_status_text', 'order_status_text', 'pay_way_text', 'order_type_text', 'verification_status_text', 'can_change_invoice_no','to_ship_btn','to_verification_btn' ])
  252. ->find();
  253. $result['send_btn'] = $result['to_ship_btn'];
  254. $result['verification_btn'] = $result['to_verification_btn'];
  255. $result['pay_time'] = $result['pay_time'] == 0 ? '未支付' : date('Y-m-d H:i:s', $result['pay_time']);
  256. $user = $result['user'];
  257. $user['avatar'] = UrlServer::getFileUrl($user['avatar']);
  258. $result['user'] = $user;
  259. foreach ($result['order_goods'] as &$order_goods) {
  260. $order_goods['goods_image'] = empty($order_goods['spec_image']) ?
  261. UrlServer::getFileUrl($order_goods['image']) : UrlServer::getFileUrl($order_goods['spec_image']);
  262. }
  263. if ($result['order_type'] == OrderEnum::TEAM_ORDER) {
  264. $team = TeamJoin::field('TJ.status,TF.status as found_status')->alias('TJ')
  265. ->where(['TJ.order_id'=>$result['id']])
  266. ->join('team_found TF', 'TF.id = TJ.team_id')
  267. ->findOrEmpty()->toArray();
  268. $result['is_team_success'] = false;
  269. if ($team) {
  270. $result['is_team_success'] = $team['found_status'] == 1 ? true : false;
  271. }
  272. }
  273. // 自提提货时间
  274. if ($result['delivery_type'] == OrderEnum::DELIVERY_TYPE_SELF && $result['verification_status']) {
  275. $result['confirm_take_time'] = date('Y-m-d H:i:s', $result['confirm_take_time']);
  276. } else {
  277. $result['confirm_take_time'] = '';
  278. }
  279. //会员优惠
  280. $result['member_amount'] = $result['member_amount'] ?? 0.00;
  281. return $result;
  282. }
  283. /**
  284. * @notes 物流信息
  285. * @param $order_id
  286. * @return array|\PDOStatement|string|Model|null
  287. * @throws \think\db\exception\DataNotFoundException
  288. * @throws \think\db\exception\ModelNotFoundException
  289. * @throws \think\exception\DbException
  290. * @author suny
  291. * @date 2021/7/14 10:24 上午
  292. */
  293. public static function shippingInfo($order_id)
  294. {
  295. $shipping = Db::name('delivery')->where('order_id', $order_id)->find();
  296. if ($shipping) {
  297. $shipping['create_time_text'] = date('Y-m-d H:i:s', $shipping['create_time']);
  298. }
  299. $shipping['traces'] = self::getShipping($order_id);
  300. return $shipping;
  301. }
  302. /**
  303. * @notes 物流轨迹
  304. * @param $order_id
  305. * @return bool|string[]
  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:24 上午
  311. */
  312. public static function getShipping($order_id)
  313. {
  314. $orderModel = new Order();
  315. $order = $orderModel->alias('o')
  316. ->field('invoice_no,shipping_name,shipping_id,o.shipping_status,o.mobile')
  317. ->join('delivery d', 'd.order_id = o.id')
  318. ->where(['o.id' => $order_id])
  319. ->find();
  320. $express = ConfigServer::get('express', 'way', '', '');
  321. $key = ConfigServer::get($express, 'appkey');
  322. $app = ConfigServer::get($express, 'appsecret');
  323. if (empty($express) || $order['shipping_status'] != 1 || empty($app) || empty($key)) {
  324. return $traces[] = ['暂无物流信息'];
  325. }
  326. //快递配置设置为快递鸟时
  327. if ($express === 'kdniao') {
  328. $expressage = (new Kdniao($app, $key, true));
  329. $shipping_field = 'codebird';
  330. } else {
  331. $expressage = (new Kd100($app, $key, true));
  332. $shipping_field = 'code100';
  333. }
  334. //快递编码
  335. $shipping_code = Db::name('express')
  336. ->where(['id' => $order['shipping_id']])
  337. ->value($shipping_field);
  338. //获取物流轨迹
  339. if (in_array(strtolower($shipping_code ), [ 'sf', 'shunfeng' ])) {
  340. if ($express === 'kdniao') {
  341. $expressage->logistics($shipping_code, $order['invoice_no'], substr($order['mobile'], -4));
  342. } else {
  343. $expressage->logistics($shipping_code, $order['invoice_no'], $order['mobile']);
  344. }
  345. }else {
  346. $expressage->logistics($shipping_code, $order['invoice_no']);
  347. }
  348. $traces = $expressage->logisticsFormat();
  349. if ($traces == false) {
  350. $traces[] = [$expressage->getError()];
  351. } else {
  352. foreach ($traces as &$item) {
  353. $item = array_values(array_unique($item));
  354. }
  355. }
  356. return $traces;
  357. }
  358. /**
  359. * @notes 获取物流
  360. * @return \think\Collection
  361. * @throws \think\db\exception\DataNotFoundException
  362. * @throws \think\db\exception\DbException
  363. * @throws \think\db\exception\ModelNotFoundException
  364. * @author suny
  365. * @date 2021/7/14 10:24 上午
  366. */
  367. public static function express()
  368. {
  369. return Express::where('del', 0)->select();
  370. }
  371. static function change_invoice_no($post)
  372. {
  373. $id = $post['order_id'];
  374. $order = Order::where(['id' => $id])->find();
  375. if (!$order) {
  376. return '订单失效';
  377. }
  378. if ($order['del'] == 1) {
  379. return '订单已删除';
  380. }
  381. if (! $order->getAttr('can_change_invoice_no')) {
  382. return '已不能修改物流单号';
  383. }
  384. //
  385. if ($post['send_type'] == 1 && empty($post['invoice_no'])) {
  386. return '请输入快递单号';
  387. }
  388. $delivery_data = [
  389. 'invoice_no' => $post['invoice_no'] ?? '',
  390. ];
  391. $shipping = Db::name('express')->where('id', $post['shipping_id'])->find();
  392. if ($post['send_type'] == 1) {
  393. $delivery_data['shipping_id'] = $post['shipping_id'];
  394. $delivery_data['shipping_name'] = $shipping['name'] ?? '';
  395. }
  396. Delivery::update($delivery_data, [ [ 'order_id', '=', $post['order_id'] ] ]);
  397. return true;
  398. }
  399. /**
  400. * @notes 发货操作
  401. * @param $data
  402. * @param $admin_id
  403. * @return bool
  404. * @throws \think\db\exception\DataNotFoundException
  405. * @throws \think\db\exception\DbException
  406. * @throws \think\db\exception\ModelNotFoundException
  407. * @throws \think\exception\DbException
  408. * @author suny
  409. * @date 2021/7/14 10:24 上午
  410. */
  411. public static function deliveryHandle($data, $admin_id)
  412. {
  413. $order_id = $data['order_id'];
  414. $order = Order::where(['del' => 0, 'id' => $order_id])->find();
  415. if ($order['shipping_status'] == 1) {
  416. return true;
  417. }
  418. $shipping = Db::name('express')->where('id', $data['shipping_id'])->find();
  419. //添加发货单
  420. $delivery_data = [
  421. 'order_id' => $order_id,
  422. 'order_sn' => $order['order_sn'],
  423. 'user_id' => $order['user_id'],
  424. 'admin_id' => $admin_id,
  425. 'consignee' => $order['consignee'],
  426. 'mobile' => $order['mobile'],
  427. 'province' => $order['province'],
  428. 'city' => $order['city'],
  429. 'district' => $order['district'],
  430. 'address' => $order['address'],
  431. 'invoice_no' => $data['invoice_no'],
  432. 'send_type' => $data['send_type'],
  433. 'create_time' => time(),
  434. ];
  435. //配送方式->快递配送
  436. if ($data['send_type'] == 1) {
  437. $delivery_data['shipping_id'] = $data['shipping_id'];
  438. $delivery_data['shipping_name'] = $shipping['name'];
  439. $delivery_data['shipping_status'] = 1;
  440. }
  441. $delivery_id = Db::name('delivery')->insertGetId($delivery_data);
  442. //更新订单下商品的发货状态
  443. $order->update_time = time();
  444. $order->shipping_time = time();
  445. $order->shipping_status = 1;
  446. $order->order_status = Order::STATUS_WAIT_RECEIVE;
  447. $order->delivery_id = $delivery_id;
  448. $order->save();
  449. //订单日志
  450. OrderLogLogic::record(
  451. OrderLogEnum::TYPE_SHOP,
  452. OrderLogEnum::SHOP_DELIVERY_ORDER,
  453. $order_id,
  454. $admin_id,
  455. OrderLogEnum::SHOP_DELIVERY_ORDER
  456. );
  457. //通知用户发货
  458. if (!empty($order['mobile'])) {
  459. event('Notice', [
  460. 'scene' => NoticeEnum::ORDER_DELIVERY_NOTICE,
  461. 'mobile' => $order['mobile'],
  462. 'params' => [
  463. 'order_id' => $order['id'],
  464. 'user_id' => $order['user_id'],
  465. 'shipping_name' => $delivery_data['shipping_name'] ?? '无需快递',
  466. 'invoice_no' => $data['invoice_no'] ?? '',
  467. ]
  468. ]);
  469. }
  470. }
  471. /**
  472. * @notes 判断是否可以发货
  473. * @param $post
  474. * @return bool|string
  475. * @throws \think\db\exception\DataNotFoundException
  476. * @throws \think\db\exception\DbException
  477. * @throws \think\db\exception\ModelNotFoundException
  478. * @author suny
  479. * @date 2021/7/14 10:24 上午
  480. */
  481. public static function checkDelivery($post)
  482. {
  483. $id = $post['order_id'];
  484. $order = Order::where(['id' => $id])->find();
  485. if (!$order) {
  486. return '订单失效';
  487. }
  488. if ($order['del'] == 1) {
  489. return '订单已删除';
  490. }
  491. if ($order['shipping_status'] == 1) {
  492. return '此订单已发货';
  493. }
  494. //
  495. if ($post['send_type'] == 1 && empty($post['invoice_no'])) {
  496. return '请输入快递单号';
  497. }
  498. if ($order['order_type'] == OrderEnum::TEAM_ORDER) {
  499. $join = TeamJoin::where(['order_id' => $order['id']])->findOrEmpty();
  500. if ($join['status'] != Team::STATUS_SUCCESS) {
  501. return '已支付的拼团订单需要等待拼团成功后才能发货';
  502. }
  503. }
  504. return true;
  505. }
  506. /**
  507. * @notes 判断是否可以取消订单
  508. * @param $post
  509. * @return bool|string
  510. * @throws \think\db\exception\DataNotFoundException
  511. * @throws \think\db\exception\DbException
  512. * @throws \think\db\exception\ModelNotFoundException
  513. * @throws \think\exception\DbException
  514. * @author suny
  515. * @date 2021/7/14 10:24 上午
  516. */
  517. public static function checkCancel($post)
  518. {
  519. $id = $post['order_id'];
  520. $order = Order::where(['id' => $id, 'del' => 0])->find();
  521. if (!$order) {
  522. return '订单失效';
  523. }
  524. if ($order['order_status'] > OrderEnum::ORDER_STATUS_DELIVERY && $order['delivery_type'] != 2) {
  525. return '此订单不可取消';
  526. }
  527. if ($order['order_status'] > OrderEnum::ORDER_STATUS_GOODS && $order['delivery_type'] == 2) {
  528. return '此订单不可取消';
  529. }
  530. if ($order['order_type'] == OrderEnum::TEAM_ORDER) {
  531. $found = Db::name('team_join')->where(['order_id' => $order['id']])->find();
  532. if ($found['status'] == Team::STATUS_WAIT_SUCCESS) {
  533. return '已支付的拼团订单需要有拼团结果才可以取消';
  534. }
  535. }
  536. return true;
  537. }
  538. /**
  539. * @notes 取消订单(返回商品规格表库存)
  540. * @param $order_id
  541. * @param $admin_id
  542. * @return bool|string
  543. * @throws \think\db\exception\DataNotFoundException
  544. * @throws \think\db\exception\ModelNotFoundException
  545. * @throws \think\exception\DbException
  546. * @throws \think\exception\PDOException
  547. * @author suny
  548. * @date 2021/7/14 10:24 上午
  549. */
  550. public static function cancel($order_id, $admin_id)
  551. {
  552. Db::startTrans();
  553. try {
  554. $order = Order::where(['id' => $order_id], ['orderGoods'])->find();
  555. // 如果是拼团订单
  556. $team_join = (new TeamJoin())->where(['order_id' => $order['id'],'status'=>TeamEnum::TEAM_STATUS_CONDUCT])->findOrEmpty()->toArray();//拼团中
  557. if ($order['order_type'] == OrderEnum::TEAM_ORDER && !empty($team_join)) {
  558. $time = time();
  559. $team_id = $team_join['team_id'];
  560. $teamJoin = (new TeamJoin())->alias('TJ')
  561. ->field(['TJ.*,O.order_sn,O.order_status,O.pay_status,O.refund_status,O.order_amount'])
  562. ->where(['team_id' => $team_id])
  563. ->join('order O', 'O.id=TJ.order_id')
  564. ->select()->toArray();
  565. TeamFound::update(['status' => TeamEnum::TEAM_STATUS_FAIL, 'team_end_time' => $time], ['id' => $team_id]);
  566. foreach ($teamJoin as $item) {
  567. TeamJoin::update(['status' => TeamEnum::TEAM_STATUS_FAIL, 'update_time' => $time], ['id' => $item['id']]);
  568. OrderRefundLogic::cancelOrder($item['order_id'], OrderLogEnum::TYPE_SHOP, $admin_id); //取消订单
  569. if ($item['pay_status'] == PayEnum::ISPAID) {
  570. $order = (new Order())->findOrEmpty($item['order_id'])->toArray();
  571. OrderRefundLogic::cancelOrderRefundUpdate($order); //更新订单状态
  572. OrderRefundLogic::refund($order, $order['order_amount'], $order['order_amount']); //订单退款
  573. }
  574. }
  575. } else {
  576. //取消订单
  577. OrderRefundLogic::cancelOrder($order_id, OrderLogEnum::TYPE_SHOP, $admin_id);
  578. //已支付的订单,取消,退款
  579. if ($order['pay_status'] == PayEnum::ISPAID) {
  580. //更新订单状态
  581. OrderRefundLogic::cancelOrderRefundUpdate($order);
  582. //订单退款
  583. OrderRefundLogic::refund($order, $order['order_amount'], $order['order_amount']);
  584. }
  585. }
  586. Db::commit();
  587. return true;
  588. } catch (\Exception $e) {
  589. Db::rollback();
  590. //增加退款失败记录
  591. OrderRefundLogic::addErrorRefund($order, $e->getMessage());
  592. return $e->getMessage();
  593. }
  594. }
  595. /**
  596. * @notes 判断是否可以删除订单
  597. * @param $post
  598. * @return bool|string
  599. * @throws \think\db\exception\DataNotFoundException
  600. * @throws \think\db\exception\DbException
  601. * @throws \think\db\exception\ModelNotFoundException
  602. * @author suny
  603. * @date 2021/7/14 10:24 上午
  604. */
  605. public static function checkDel($post)
  606. {
  607. $id = $post['order_id'];
  608. $order = Order::where(['id' => $id])->find();
  609. if (!$order) {
  610. return '订单失效';
  611. }
  612. if ($order['del'] == 1) {
  613. return '订单已删除';
  614. }
  615. if ($order['order_status'] != OrderEnum::ORDER_STATUS_DOWN) {
  616. return '此订单不可删除';
  617. }
  618. return true;
  619. }
  620. /**
  621. * @notes 删除已取消的订单
  622. * @param $order_id
  623. * @param $admin_id
  624. * @throws \think\db\exception\DataNotFoundException
  625. * @throws \think\db\exception\DbException
  626. * @throws \think\db\exception\ModelNotFoundException
  627. * @author suny
  628. * @date 2021/7/14 10:24 上午
  629. */
  630. public static function del($order_id, $admin_id)
  631. {
  632. $order = Order::where(['delete' => 0, 'id' => $order_id])->find();
  633. $order->save(['delete' => 1, 'update_time' => time()]);
  634. //订单日志
  635. OrderLogLogic::record(
  636. OrderLogEnum::TYPE_SHOP,
  637. OrderLogEnum::SHOP_DEL_ORDER,
  638. $order_id,
  639. $admin_id,
  640. OrderLogEnum::SHOP_DEL_ORDER
  641. );
  642. }
  643. /**
  644. * @notes 确认收货
  645. * @param $order_id
  646. * @param $admin_id
  647. * @throws \think\db\exception\DataNotFoundException
  648. * @throws \think\db\exception\DbException
  649. * @throws \think\db\exception\ModelNotFoundException
  650. * @author suny
  651. * @date 2021/7/14 10:25 上午
  652. */
  653. public static function confirm($order_id, $admin_id)
  654. {
  655. $order = Order::where(['del' => 0, 'id' => $order_id])->find();
  656. $order->order_status = Order::STATUS_FINISH;
  657. $order->update_time = time();
  658. $order->confirm_take_time = time();
  659. $order->save();
  660. //订单日志
  661. OrderLogLogic::record(
  662. OrderLogEnum::TYPE_SHOP,
  663. OrderLogEnum::SHOP_CONFIRM_ORDER,
  664. $order_id,
  665. $admin_id,
  666. OrderLogEnum::SHOP_CONFIRM_ORDER
  667. );
  668. }
  669. /**
  670. * @notes 订单备注
  671. * @param $post
  672. * @param string $type
  673. * @return Order|array|Model
  674. * @author suny
  675. * @date 2021/7/14 10:25 上午
  676. */
  677. public static function remarks($post, $type = "get")
  678. {
  679. if ($type === 'get') {
  680. return Order::field('id,order_remarks')
  681. ->where(['id' => $post['id']])
  682. ->findOrEmpty();
  683. } else {
  684. return Order::where(['id' => $post['id']])
  685. ->update(['order_remarks' => $post['order_remarks']]);
  686. }
  687. }
  688. /**
  689. * @notes 打印订单
  690. * @param $id
  691. * @param $shop_id
  692. * @return string|void
  693. * @author 段誉
  694. * @date 2022/1/20 11:24
  695. */
  696. public static function orderPrint($id, $shop_id)
  697. {
  698. try {
  699. //打印机配置
  700. $config = PrinterConfig::where(['status' => 1, 'shop_id' => $shop_id])->findOrEmpty();
  701. //打印机列表
  702. $printers = Printer::where([
  703. 'shop_id' => $shop_id,
  704. 'config_id' => $config['id'],
  705. 'del' => 0,
  706. 'status' => 1
  707. ])->select()->toArray();
  708. if ($config->isEmpty() || empty($printers)) {
  709. throw new Exception('请先配置打印机');
  710. }
  711. // 订单信息
  712. $order = self::getPrintOrder($id);
  713. $yly_print = new YlyPrinter($config['client_id'], $config['client_secret'], $shop_id);
  714. // 模板
  715. $template = PrinterLogic::getPrinterTpl($shop_id);
  716. $yly_print->ylyPrint($printers, $order, $template);
  717. return true;
  718. } catch (\Exception $e) {
  719. $msg = json_decode($e->getMessage(), true);
  720. if ($msg && isset($msg['error'])) {
  721. return '易联云:' . $msg['error_description'];
  722. }
  723. if (18 === $e->getCode()) {
  724. Cache::rm('yly_access_token' . $shop_id);
  725. Cache::rm('yly_refresh_token' . $shop_id);
  726. }
  727. return '易联云:' . $e->getMessage();
  728. }
  729. }
  730. /**
  731. * @notes 获取打印订单
  732. * @param $id
  733. * @return array
  734. * @author 段誉
  735. * @date 2022/1/21 9:58
  736. */
  737. public static function getPrintOrder($id)
  738. {
  739. return (new Order())
  740. ->with(['user', 'order_goods'])
  741. ->where('id', $id)
  742. ->append(['delivery_address'])
  743. ->findOrEmpty()->toArray();
  744. }
  745. /**
  746. * @notes 获取地址
  747. * @param $post
  748. * @return array
  749. * @throws \think\db\exception\DataNotFoundException
  750. * @throws \think\db\exception\DbException
  751. * @throws \think\db\exception\ModelNotFoundException
  752. * @author suny
  753. * @date 2021/7/14 10:25 上午
  754. */
  755. public static function change_address($post)
  756. {
  757. $order = new Order();
  758. $info = $order->where(['id' => $post['id']])
  759. ->field(['id', 'province', 'city', 'district', 'address','consignee','mobile'])
  760. ->find()->toArray();
  761. $info = json_encode($info, JSON_UNESCAPED_UNICODE);
  762. $address_tree = json_encode(self::getAddressTree(), JSON_UNESCAPED_UNICODE);
  763. $data = [
  764. 'info' => $info,
  765. 'address_tree' => $address_tree
  766. ];
  767. return $data;
  768. }
  769. /**
  770. * @notes 获取城市信息
  771. * @return \think\Collection
  772. * @throws \think\db\exception\DataNotFoundException
  773. * @throws \think\db\exception\DbException
  774. * @throws \think\db\exception\ModelNotFoundException
  775. * @author suny
  776. * @date 2021/7/14 10:25 上午
  777. */
  778. public static function getAddressTree()
  779. {
  780. $DevRegion = new DevRegion();
  781. $lists = $DevRegion
  782. ->field(['name', 'id', 'parent_id', 'level'])
  783. ->where('level', '>', 0)
  784. ->select();
  785. return $lists;
  786. }
  787. /**
  788. * @notes 修改地址
  789. * @param $post
  790. * @author suny
  791. * @date 2021/7/14 10:27 上午
  792. */
  793. public static function change_address_post($post)
  794. {
  795. $order = new Order();
  796. $data = [
  797. 'province' => $post['province'],
  798. 'city' => $post['city'],
  799. 'district' => $post['district'],
  800. 'address' => $post['address'],
  801. 'consignee' => $post['consignee'],
  802. 'mobile' => $post['mobile'],
  803. ];
  804. $order->where(['id' => $post['order_id']])->update($data);
  805. }
  806. /**
  807. * @notes 获取各列表数量
  808. * @param $shop_id
  809. * @return int
  810. * @author suny
  811. * @date 2021/7/14 10:27 上午
  812. */
  813. public static function getAll($shop_id)
  814. {
  815. $data = input();
  816. unset($data['type']);
  817. return Order::alias('o')
  818. ->join('shop s', 's.id = o.shop_id')
  819. ->join('user u', 'u.id = o.user_id')
  820. ->join('order_goods g', 'g.order_id = o.id')
  821. ->with(['order_goods', 'user', 'shop'])
  822. ->where(static::getWhere($data))
  823. ->where('o.shop_id', $shop_id)
  824. ->where('o.del', 0)
  825. ->where('o.delete', 0)
  826. ->group('o.id')
  827. ->count();
  828. }
  829. /**
  830. * @notes 拼装统计信息
  831. * @param $order_status
  832. * @return array
  833. * @author suny
  834. * @date 2021/7/13 3:36 下午
  835. */
  836. public static function getStat($order_status, $shop_id)
  837. {
  838. $data = input();
  839. unset($data['type']);
  840. $result = [];
  841. foreach ($order_status as $status => $title) {
  842. $num = Order::alias('o')
  843. ->join('shop s', 's.id = o.shop_id')
  844. ->join('user u', 'u.id = o.user_id')
  845. ->join('order_goods g', 'g.order_id = o.id')
  846. ->with(['order_goods', 'user', 'shop'])
  847. ->where(static::getWhere($data))
  848. ->where('o.order_status', $status)
  849. ->where('o.shop_id', $shop_id)
  850. ->where('o.del', 0)
  851. ->where('o.delete', 0)
  852. ->group('o.id')
  853. ->count();
  854. $result[] = [
  855. 'title' => $title,
  856. 'status' => $status,
  857. 'count' => $num,
  858. ];
  859. }
  860. return $result;
  861. }
  862. /**
  863. * @notes 虚拟商品发货
  864. * @param $post
  865. * @param $admin_id
  866. * @return bool|void
  867. * @throws \think\db\exception\DataNotFoundException
  868. * @throws \think\db\exception\DbException
  869. * @throws \think\db\exception\ModelNotFoundException
  870. * @author 段誉
  871. * @date 2022/4/7 17:55
  872. */
  873. public static function virtualDelivery($post, $admin_id)
  874. {
  875. try {
  876. $order_id = $post['order_id'];
  877. $order = Order::with(['order_goods'])->where(['del' => 0, 'id' => $order_id])->find();
  878. // 更新发货订单信息
  879. $result = GoodsVirtualLogic::shopSelfDelivery($order_id, $post['delivery_content']);
  880. if (true !== $result) {
  881. throw new \Exception($result);
  882. }
  883. //订单日志
  884. OrderLogLogic::record(
  885. OrderLogEnum::TYPE_SHOP,
  886. OrderLogEnum::SHOP_DELIVERY_ORDER,
  887. $order_id,
  888. $admin_id,
  889. OrderLogEnum::SHOP_DELIVERY_ORDER
  890. );
  891. //通知用户发货
  892. if (!empty($order['mobile'])) {
  893. event('Notice', [
  894. 'scene' => NoticeEnum::ORDER_DELIVERY_NOTICE,
  895. 'mobile' => $order['mobile'],
  896. 'params' => [
  897. 'order_id' => $order['id'],
  898. 'user_id' => $order['user_id'],
  899. 'shipping_name' => '无需快递',
  900. 'invoice_no' => '',
  901. ]
  902. ]);
  903. }
  904. return true;
  905. } catch (\Exception $e) {
  906. self::$error = $e->getMessage();
  907. return false;
  908. }
  909. }
  910. /**
  911. * @notes 导出Excel
  912. * @param array $condition
  913. * @return array|false
  914. * @author 段誉
  915. * @date 2022/4/24 10:10
  916. */
  917. public static function export($condition = [])
  918. {
  919. try {
  920. $field = 'o.*,order_status as order_status_text,pay_way as pay_way_text,
  921. o.delivery_type as delivery_type_text,order_type as order_type_text,
  922. u.nickname,s.name as shop_name';
  923. $lists = Order::alias('o')
  924. ->join('shop s', 's.id = o.shop_id')
  925. ->join('user u', 'u.id = o.user_id')
  926. ->join('order_goods g', 'g.order_id = o.id')
  927. ->with(['order_goods'])
  928. ->field($field)
  929. ->where($condition)
  930. ->append(['delivery_address', 'pay_status_text', 'order_source_text'])
  931. ->order('o.id desc')
  932. ->group('o.id')
  933. ->select()
  934. ->toArray();
  935. foreach ($lists as &$item) {
  936. $orderGoodsList = [];
  937. $goodsItemList = [];
  938. $goodsPriceList = [];
  939. $goodsNumList = [];
  940. foreach ($item['order_goods'] as $good) {
  941. $orderGoodsList[] = $good['goods_name'];
  942. $goodsItemList[] = $good['spec_value'];
  943. $goodsPriceList[] = $good['goods_price'];
  944. $goodsNumList[] = $good['goods_num'];
  945. }
  946. $item['order_goods_list'] = implode(';', $orderGoodsList);
  947. $item['goods_item_list'] = implode(';', $goodsItemList);
  948. $item['goods_price_list'] = implode(';', $goodsPriceList);
  949. $item['goods_num_list'] = implode(';', $goodsNumList);
  950. }
  951. $excelFields = [
  952. 'order_sn' => '订单编号',
  953. 'order_type_text' => '订单类型',
  954. 'nickname' => '用户名称',
  955. 'order_goods_list' => '商品信息',
  956. 'goods_item_list' => '规格',
  957. 'goods_price_list' => '商品价格',
  958. 'goods_num_list' => '商品数量',
  959. 'order_amount' => '实付金额',
  960. 'consignee' => '收货人',
  961. 'mobile' => '收货人手机',
  962. 'delivery_address' => '收货地址',
  963. 'pay_status_text' => '支付状态',
  964. 'order_status_text' => '订单状态',
  965. 'create_time' => '下单时间',
  966. ];
  967. $export = new ExportExcelServer();
  968. $export->setFileName('订单');
  969. $result = $export->createExcel($excelFields, $lists);
  970. return ['url' => $result];
  971. } catch (\Exception $e) {
  972. self::$error = $e->getMessage();
  973. return false;
  974. }
  975. }
  976. /**
  977. * @notes 提货核销
  978. * @param $params
  979. * @param $adminInfo
  980. * @return bool
  981. * @author 段誉
  982. * @date 2022/11/2 14:58
  983. */
  984. public static function verification($params, $adminInfo)
  985. {
  986. Db::startTrans();
  987. try {
  988. $order = Order::find($params['order_id']);
  989. //添加核销记录
  990. Verification::create([
  991. 'order_id' => $order['id'],
  992. 'shop_id' => $order['shop_id'],
  993. 'handle_id' => $adminInfo['id'],
  994. 'verification_scene' => VerificationEnum::TYPE_SHOP,
  995. 'snapshot' => json_encode([
  996. 'sn' => $adminInfo['account'],
  997. 'name' => $adminInfo['name']
  998. ]),
  999. ]);
  1000. //更新订单状态
  1001. $order->order_status = OrderEnum::ORDER_STATUS_COMPLETE;
  1002. $order->verification_status = OrderEnum::WRITTEN_OFF;
  1003. $order->confirm_take_time = time();
  1004. $order->save();
  1005. //订单日志
  1006. OrderLogLogic::record(
  1007. OrderLogEnum::TYPE_SHOP,
  1008. OrderLogEnum::SHOP_VERIFICATION,
  1009. $order['id'],
  1010. $adminInfo['id'],
  1011. OrderLogEnum::SHOP_VERIFICATION
  1012. );
  1013. Db::commit();
  1014. return true;
  1015. } catch (\Exception $e) {
  1016. Db::rollback();
  1017. self::$error = $e->getMessage();
  1018. return false;
  1019. }
  1020. }
  1021. /**
  1022. * @notes 确认付款
  1023. * @param $order_id
  1024. * @param $adminInfo
  1025. * @return string|true
  1026. * @author ljj
  1027. * @date 2024/7/19 下午6:32
  1028. */
  1029. public static function confirmPay($order_id, $adminInfo)
  1030. {
  1031. Db::startTrans();
  1032. try {
  1033. $order = Order::where(['id' => $order_id])->findOrEmpty()->toArray();
  1034. $result = PayNotifyLogic::handle('order', $order['order_sn']);
  1035. if (true !== $result) {
  1036. throw new \Exception($result);
  1037. }
  1038. Db::commit();
  1039. return true;
  1040. } catch (\Exception $e) {
  1041. Db::rollback();
  1042. return $e->getMessage();
  1043. }
  1044. }
  1045. }