Order.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. <?php
  2. namespace app\data\controller\api\auth;
  3. use app\data\controller\api\Auth;
  4. use app\data\service\GoodsService;
  5. use app\data\service\OrderService;
  6. use app\data\service\PaymentService;
  7. use app\data\service\TruckService;
  8. use app\data\service\UserService;
  9. use think\admin\extend\CodeExtend;
  10. use think\exception\HttpResponseException;
  11. /**
  12. * 用户订单数据接口
  13. * Class Order
  14. * @package app\data\controller\api\auth
  15. */
  16. class Order extends Auth
  17. {
  18. /**
  19. * 控制器初始化
  20. */
  21. protected function initialize()
  22. {
  23. parent::initialize();
  24. if (empty($this->user['status'])) {
  25. $this->error('账户已被冻结,不能操作订单数据哦!');
  26. }
  27. }
  28. /**
  29. * 获取订单列表
  30. * @throws \think\db\exception\DataNotFoundException
  31. * @throws \think\db\exception\DbException
  32. * @throws \think\db\exception\ModelNotFoundException
  33. */
  34. public function get()
  35. {
  36. $map = ['uid' => $this->uuid, 'deleted' => 0];
  37. $query = $this->_query('ShopOrder')->in('status')->equal('order_no');
  38. $result = $query->where($map)->order('id desc')->page(true, false, false, 20);
  39. if (count($result['list']) > 0) OrderService::instance()->buildItemData($result['list']);
  40. $this->success('获取订单数据成功!', $result);
  41. }
  42. /**
  43. * 用户创建订单
  44. * @throws \think\db\exception\DataNotFoundException
  45. * @throws \think\db\exception\DbException
  46. * @throws \think\db\exception\ModelNotFoundException
  47. */
  48. public function add()
  49. {
  50. // 商品规则
  51. $rules = $this->request->post('items', '');
  52. if (empty($rules)) $this->error('商品不能为空');
  53. // 订单数据
  54. [$codes, $items, $truckType] = [[], [], -1];
  55. $order = ['uid' => $this->uuid];
  56. $order['order_no'] = CodeExtend::uniqidDate(18, 'N');
  57. // 推荐人处理
  58. $order['from'] = input('from_uid', $this->user['pid1']);
  59. if ($order['from'] == $this->uuid) $order['from'] = 0;
  60. if ($order['from'] > 0) {
  61. $map = ['id' => $order['from'], 'status' => 1];
  62. $fromer = $this->app->db->name('DataUser')->where($map)->find();
  63. if (empty($fromer)) $this->error('未找到推荐人');
  64. }
  65. foreach (explode('||', $rules) as $rule) {
  66. [$code, $spec, $count] = explode('@', $rule);
  67. // 商品信息检查
  68. $goodsInfo = $this->app->db->name('ShopGoods')->where(['code' => $code, 'status' => 1, 'deleted' => 0])->find();
  69. $goodsItem = $this->app->db->name('ShopGoodsItem')->where(['goods_code' => $code, 'goods_spec' => $spec, 'status' => 1])->find();
  70. if (empty($goodsInfo)) $this->error('商品数据查询异常');
  71. if (empty($goodsItem)) $this->error('商品规格查询异常');
  72. // 商品类型检查
  73. if ($truckType < 0) $truckType = $goodsInfo['truck_type'];
  74. if ($truckType !== $goodsInfo['truck_type']) $this->error('实物与虚拟不能混合下单!');
  75. // 限制购买数量
  76. if (isset($goods['limit_max_num']) && $goods['limit_max_num'] > 0) {
  77. $map = [['a.status', 'in', [2, 3, 4, 5]], ['b.goods_code', '=', $goods['code']], ['a.uid', '=', $this->uuid]];
  78. $buys = $this->app->db->name('StoreOrder')->alias('a')->join('store_order_item b', 'a.order_no=b.order_no')->where($map)->sum('b.stock_sales');
  79. if ($buys + $count > $goods['limit_max_num']) $this->error('超过限购数量');
  80. }
  81. // 限制购买身份
  82. if ($goodsInfo['limit_low_vip'] > $this->user['vip_number']) {
  83. $this->error('用户等级不够');
  84. }
  85. // 商品库存检查
  86. if ($goodsItem['stock_sales'] + $count > $goodsItem['stock_total']) {
  87. $this->error('商品库存不足');
  88. }
  89. // 商品折扣处理
  90. [$discountId, $discountRate] = [0, 100];
  91. if ($goodsInfo['discount_id'] > 0) {
  92. $map = ['status' => 1, 'deleted' => 0, 'id' => $goodsInfo['discount_id']];
  93. if ($items = $this->app->db->name('DataUserDiscount')->where($map)->value('items')) {
  94. foreach (json_decode($items, true) as $vo) if ($vo['level'] == $this->user['vip_number']) {
  95. [$discountId, $discountRate] = [$goodsInfo['discount_id'], $vo['discount']];
  96. }
  97. }
  98. }
  99. // 订单详情处理
  100. $items[] = [
  101. 'uid' => $order['uid'],
  102. 'order_no' => $order['order_no'],
  103. // 商品信息字段
  104. 'goods_name' => $goodsInfo['name'],
  105. 'goods_cover' => $goodsInfo['cover'],
  106. 'goods_sku' => $goodsItem['goods_sku'],
  107. 'goods_code' => $goodsItem['goods_code'],
  108. 'goods_spec' => $goodsItem['goods_spec'],
  109. // 库存数量处理
  110. 'stock_sales' => $count,
  111. // 快递发货数据
  112. 'truck_type' => $goodsInfo['truck_type'],
  113. 'truck_code' => $goodsInfo['truck_code'],
  114. 'truck_count' => $goodsItem['number_express'] * $count,
  115. // 商品费用字段
  116. 'price_market' => $goodsItem['price_market'],
  117. 'price_selling' => $goodsItem['price_selling'],
  118. 'total_market' => $goodsItem['price_market'] * $count,
  119. 'total_selling' => $goodsItem['price_selling'] * $count,
  120. // 奖励金额积分
  121. 'reward_balance' => $goodsItem['reward_balance'] * $count,
  122. 'reward_integral' => $goodsItem['reward_integral'] * $count,
  123. // 绑定用户等级
  124. 'vip_name' => $this->user['vip_name'],
  125. 'vip_number' => $this->user['vip_number'],
  126. // 是否入会礼包
  127. 'vip_entry' => $goodsInfo['vip_entry'],
  128. // 等级优惠方案
  129. 'discount_id' => $discountId,
  130. 'discount_rate' => $discountRate,
  131. 'discount_amount' => $discountRate * $goodsItem['price_selling'] * $count / 100,
  132. ];
  133. }
  134. try {
  135. // 订单发货类型
  136. $order['truck_type'] = $truckType;
  137. $order['status'] = $truckType ? 2 : 1;
  138. // 统计商品数量
  139. $order['number_goods'] = array_sum(array_column($items, 'stock_sales'));
  140. // 统计商品金额
  141. $order['amount_goods'] = array_sum(array_column($items, 'total_selling'));
  142. // 优惠后的金额
  143. $order['amount_discount'] = array_sum(array_column($items, 'discount_amount'));
  144. // 订单随机免减
  145. $order['amount_reduct'] = OrderService::instance()->getReduct();
  146. if ($order['amount_reduct'] > $order['amount_goods']) {
  147. $order['amount_reduct'] = $order['amount_goods'];
  148. }
  149. // 统计订单金额
  150. $order['amount_real'] = $order['amount_discount'] - $order['amount_reduct'];
  151. $order['amount_total'] = $order['amount_goods'];
  152. // 写入商品数据
  153. $this->app->db->transaction(function () use ($order, $items) {
  154. $this->app->db->name('ShopOrder')->insert($order);
  155. $this->app->db->name('ShopOrderItem')->insertAll($items);
  156. });
  157. // 同步商品库存销量
  158. foreach ($codes as $code) GoodsService::instance()->syncStock($code);
  159. // 触发订单创建事件
  160. $this->app->event->trigger('ShopOrderCreate', $order['order_no']);
  161. // 组装订单商品数据
  162. $order['items'] = $items;
  163. // 返回处理成功数据
  164. $this->success('商品下单成功', $order);
  165. } catch (HttpResponseException $exception) {
  166. throw $exception;
  167. } catch (\Exception $exception) {
  168. $this->error("商品下单失败,{$exception->getMessage()}");
  169. }
  170. }
  171. /**
  172. * 模拟计算订单运费
  173. * @throws \think\db\exception\DataNotFoundException
  174. * @throws \think\db\exception\DbException
  175. * @throws \think\db\exception\ModelNotFoundException
  176. */
  177. public function express()
  178. {
  179. $data = $this->_vali([
  180. 'code.require' => '地址不能为空',
  181. 'order_no.require' => '单号不能为空',
  182. ]);
  183. // 用户收货地址
  184. $map = ['uid' => $this->uuid, 'code' => $data['code']];
  185. $addr = $this->app->db->name('DataUserAddress')->where($map)->find();
  186. if (empty($addr)) $this->error('收货地址异常');
  187. // 订单状态检查
  188. $map = ['uid' => $this->uuid, 'order_no' => $data['order_no']];
  189. $tCount = $this->app->db->name('ShopOrderItem')->where($map)->sum('truck_count');
  190. // 根据地址计算运费
  191. $map = ['status' => 1, 'deleted' => 0, 'order_no' => $data['order_no']];
  192. $tCode = $this->app->db->name('ShopOrderItem')->where($map)->column('truck_code');
  193. [$amount, , , $remark] = TruckService::instance()->amount($tCode, $addr['province'], $addr['city'], $tCount);
  194. $this->success('计算运费成功', ['amount' => $amount, 'remark' => $remark]);
  195. }
  196. /**
  197. * 订单信息完成
  198. * @throws \think\db\exception\DataNotFoundException
  199. * @throws \think\db\exception\DbException
  200. * @throws \think\db\exception\ModelNotFoundException
  201. */
  202. public function perfect()
  203. {
  204. $data = $this->_vali([
  205. 'code.require' => '地址不能为空',
  206. 'order_no.require' => '单号不能为空',
  207. ]);
  208. // 用户收货地址
  209. $map = ['uid' => $this->uuid, 'code' => $data['code'], 'deleted' => 0];
  210. $addr = $this->app->db->name('DataUserAddress')->where($map)->find();
  211. if (empty($addr)) $this->error('收货地址异常');
  212. // 订单状态检查
  213. $map = ['uid' => $this->uuid, 'order_no' => $data['order_no']];
  214. $order = $this->app->db->name('ShopOrder')->where($map)->whereIn('status', [1, 2])->find();
  215. $tCount = $this->app->db->name('ShopOrderItem')->where($map)->sum('truck_count');
  216. if (empty($order)) $this->error('不能修改地址');
  217. // 根据地址计算运费
  218. $map = ['status' => 1, 'deleted' => 0, 'order_no' => $data['order_no']];
  219. $tCodes = $this->app->db->name('ShopOrderItem')->where($map)->column('truck_code');
  220. [$amount, $tCount, $tCode, $remark] = TruckService::instance()->amount($tCodes, $addr['province'], $addr['city'], $tCount);
  221. // 创建订单发货信息
  222. $express = [
  223. 'template_code' => $tCode, 'template_count' => $tCount, 'uid' => $this->uuid,
  224. 'template_remark' => $remark, 'template_amount' => $amount, 'status' => 1,
  225. ];
  226. $express['order_no'] = $data['order_no'];
  227. $express['address_code'] = $data['code'];
  228. $express['address_name'] = $addr['name'];
  229. $express['address_phone'] = $addr['phone'];
  230. $express['address_idcode'] = $addr['idcode'];
  231. $express['address_province'] = $addr['province'];
  232. $express['address_city'] = $addr['city'];
  233. $express['address_area'] = $addr['area'];
  234. $express['address_content'] = $addr['address'];
  235. $express['address_datetime'] = date('Y-m-d H:i:s');
  236. data_save('ShopOrderSend', $express, 'order_no');
  237. // 组装更新订单数据
  238. $update = ['status' => 2, 'amount_express' => $express['template_amount']];
  239. // 重新计算订单金额
  240. $update['amount_real'] = $order['amount_discount'] + $amount - $order['amount_reduct'];
  241. $update['amount_total'] = $order['amount_goods'] + $amount;
  242. // 支付金额不能为零
  243. if ($update['amount_real'] <= 0) $update['amount_real'] = 0.00;
  244. if ($update['amount_total'] <= 0) $update['amount_total'] = 0.00;
  245. // 更新用户订单数据
  246. $map = ['uid' => $this->uuid, 'order_no' => $data['order_no']];
  247. if ($this->app->db->name('ShopOrder')->where($map)->update($update) !== false) {
  248. // 触发订单确认事件
  249. $this->app->event->trigger('ShopOrderPerfect', $order['order_no']);
  250. // 返回处理成功数据
  251. $this->success('订单确认成功', ['order_no' => $order['order_no']]);
  252. } else {
  253. $this->error('订单确认失败');
  254. }
  255. }
  256. /**
  257. * 获取订单支付状态
  258. * @throws \think\db\exception\DataNotFoundException
  259. * @throws \think\db\exception\DbException
  260. * @throws \think\db\exception\ModelNotFoundException
  261. */
  262. public function payment()
  263. {
  264. $data = $this->_vali([
  265. 'order_no.require' => '单号不能为空',
  266. 'payment_code.require' => '参数不能为空',
  267. 'payment_back.default' => '', # 支付回跳地址
  268. ]);
  269. $map = ['order_no' => $data['order_no']];
  270. $order = $this->app->db->name('ShopOrder')->where($map)->find();
  271. if (empty($order)) $this->error('读取订单失败');
  272. if ($order['status'] != 2) $this->error('不能发起支付');
  273. if ($order['payment_status'] > 0) $this->error('已经完成支付');
  274. try {
  275. $openid = '';
  276. if (in_array($this->type, [UserService::APITYPE_WXAPP, UserService::APITYPE_WECHAT])) {
  277. $openid = $this->user[UserService::TYPES[$this->type]['auth']] ?? '';
  278. if (empty($openid)) $this->error("无法创建支付");
  279. }
  280. // 返回订单数据及支付发起参数
  281. $type = $order['amount_real'] <= 0 ? 'empty' : $data['payment_code'];
  282. $param = PaymentService::instance($type)->create($openid, $order['order_no'], $order['amount_real'], '商城订单支付', '', $data['payment_back']);
  283. $order = $this->app->db->name('ShopOrder')->where($map)->find() ?: new \stdClass();
  284. $this->success('获取支付参数', ['order' => $order, 'param' => $param]);
  285. } catch (HttpResponseException $exception) {
  286. throw $exception;
  287. } catch (\Exception $exception) {
  288. $this->error($exception->getMessage());
  289. }
  290. }
  291. /**
  292. * 主动取消未支付的订单
  293. * @throws \think\db\exception\DataNotFoundException
  294. * @throws \think\db\exception\DbException
  295. * @throws \think\db\exception\ModelNotFoundException
  296. */
  297. public function cancel()
  298. {
  299. $map = $this->_vali([
  300. 'uid.value' => $this->uuid,
  301. 'order_no.require' => '单号不能为空',
  302. ]);
  303. $order = $this->app->db->name('ShopOrder')->where($map)->find();
  304. if (empty($order)) $this->error('读取订单失败');
  305. if (in_array($order['status'], [1, 2])) {
  306. $result = $this->app->db->name('ShopOrder')->where($map)->update([
  307. 'status' => 0,
  308. 'cancel_status' => 1,
  309. 'cancel_remark' => '用户主动取消订单!',
  310. 'cancel_datetime' => date('Y-m-d H:i:s'),
  311. ]);
  312. if ($result !== false && OrderService::instance()->syncStock($order['order_no'])) {
  313. // 触发订单取消事件
  314. $this->app->event->trigger('ShopOrderCancel', $order['order_no']);
  315. // 返回处理成功数据
  316. $this->success('订单取消成功');
  317. } else {
  318. $this->error('订单取消失败');
  319. }
  320. } else {
  321. $this->error('订单不可取消');
  322. }
  323. }
  324. /**
  325. * 用户主动删除已取消的订单
  326. * @throws \think\db\exception\DataNotFoundException
  327. * @throws \think\db\exception\DbException
  328. * @throws \think\db\exception\ModelNotFoundException
  329. */
  330. public function remove()
  331. {
  332. $map = $this->_vali([
  333. 'uid.value' => $this->uuid,
  334. 'order_no.require' => '单号不能为空',
  335. ]);
  336. $order = $this->app->db->name('ShopOrder')->where($map)->find();
  337. if (empty($order)) $this->error('读取订单失败');
  338. if (in_array($order['status'], [0])) {
  339. $result = $this->app->db->name('ShopOrder')->where($map)->update([
  340. 'status' => 0,
  341. 'deleted' => 1,
  342. 'deleted_remark' => '用户主动删除订单!',
  343. 'deleted_datetime' => date('Y-m-d H:i:s'),
  344. ]);
  345. if ($result !== false) {
  346. // 触发订单删除事件
  347. $this->app->event->trigger('ShopOrderRemove', $order['order_no']);
  348. // 返回处理成功数据
  349. $this->success('订单删除成功');
  350. } else {
  351. $this->error('订单删除失败');
  352. }
  353. } else {
  354. $this->error('订单不可删除');
  355. }
  356. }
  357. /**
  358. * 订单确认收货
  359. * @throws \think\db\exception\DataNotFoundException
  360. * @throws \think\db\exception\DbException
  361. * @throws \think\db\exception\ModelNotFoundException
  362. */
  363. public function confirm()
  364. {
  365. $map = $this->_vali([
  366. 'uid.value' => $this->uuid,
  367. 'order_no.require' => '单号不能为空',
  368. ]);
  369. $order = $this->app->db->name('ShopOrder')->where($map)->find();
  370. if (empty($order)) $this->error('读取订单失败');
  371. if (in_array($order['status'], [4])) {
  372. if ($this->app->db->name('ShopOrder')->where($map)->update(['status' => 5]) !== false) {
  373. // 触发订单确认事件
  374. $this->app->event->trigger('ShopOrderConfirm', $order['order_no']);
  375. // 返回处理成功数据
  376. $this->success('订单确认成功');
  377. } else {
  378. $this->error('订单确认失败');
  379. }
  380. } else {
  381. $this->error('订单确认失败');
  382. }
  383. }
  384. /**
  385. * 订单状态统计
  386. * @throws \think\db\exception\DataNotFoundException
  387. * @throws \think\db\exception\DbException
  388. * @throws \think\db\exception\ModelNotFoundException
  389. */
  390. public function total()
  391. {
  392. $map = ['uid' => $this->uuid, 'deleted' => 0];
  393. $data = ['t0' => 0, 't1' => 0, 't2' => 0, 't3' => 0, 't4' => 0, 't5' => 0];
  394. $query = $this->app->db->name('ShopOrder')->fieldRaw('status,count(1) count');
  395. $query->where($map)->group('status')->select()->each(function ($item) use (&$data) {
  396. $data["t{$item['status']}"] = $item['count'];
  397. });
  398. $this->success('获取统计成功', $data);
  399. }
  400. /**
  401. * 物流追踪查询
  402. */
  403. public function track()
  404. {
  405. try {
  406. $data = $this->_vali([
  407. 'code.require' => '快递不能为空',
  408. 'number.require' => '单号不能为空',
  409. ]);
  410. $result = TruckService::instance()->query($data['code'], $data['number']);
  411. empty($result['code']) ? $this->error($result['info']) : $this->success('快递追踪信息', $result);
  412. } catch (HttpResponseException $exception) {
  413. throw $exception;
  414. } catch (\Exception $exception) {
  415. $this->error($exception->getMessage());
  416. }
  417. }
  418. }