CashierOrderCreate.php 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705
  1. <?php
  2. /**
  3. * Niushop商城系统 - 团队十年电商经验汇集巨献!
  4. * =========================================================
  5. * Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
  6. * ----------------------------------------------
  7. * 官方网址: https://www.niushop.com
  8. * =========================================================
  9. */
  10. namespace addon\cashier\model\order;
  11. use addon\cardservice\model\MemberCard;
  12. use app\model\BaseModel;
  13. use app\model\order\OrderCreateTool;
  14. use app\model\order\OrderLog;
  15. use app\model\system\Pay;
  16. /**
  17. * 订单创建(普通订单)
  18. *
  19. * @author Administrator
  20. *
  21. */
  22. class CashierOrderCreate extends BaseModel
  23. {
  24. use OrderCreateTool;
  25. private $error = 0; //是否有错误
  26. private $error_printf = [];
  27. public $pay_type = 'ONLINE_PAY';
  28. public $trade_config = [];//配送方式的配置
  29. public function create($calculate_data)
  30. {
  31. $calculate_data = $this->calculate($calculate_data);
  32. if ($calculate_data[ 'code' ] < 0)
  33. return $calculate_data;
  34. if (isset($calculate_data[ 'code' ]) && $calculate_data[ 'code' ] < 0)
  35. return $calculate_data;
  36. $check_error = $this->checkError();
  37. if ($check_error[ 'code' ] < 0) {
  38. return $check_error;
  39. }
  40. $calculate_data = $calculate_data[ 'data' ];
  41. model('order')->startTrans();
  42. try {
  43. $member_id = $calculate_data[ 'member_id' ] ?? 0;
  44. $pay_model = new Pay();
  45. $out_trade_no = $pay_model->createOutTradeNo($member_id);
  46. $site_id = $calculate_data[ 'site_id' ];
  47. $order_no = $this->createOrderNo($site_id, $member_id);
  48. $site_info = $calculate_data[ 'site_info' ];
  49. $order_type = 5;//收银订单
  50. $order_name = $calculate_data[ 'order_name' ] ?? '';
  51. $member_info = $calculate_data[ 'member_info' ] ?? [];
  52. $nickname = $member_info[ 'nickname' ] ?? '';//会员昵称
  53. $headimg = $member_info[ 'headimg' ] ?? '';
  54. $mobile = $member_info[ 'mobile' ] ?? '';
  55. $nickname = !empty($nickname) ? $nickname : $mobile;
  56. $goods_list = $calculate_data[ 'goods_list' ] ?? [];
  57. $pay_money = $calculate_data[ 'pay_money' ] ?? 0;
  58. $order_status = '0';//订单状态
  59. $store_id = $calculate_data[ 'store_id' ] ?? 0;
  60. $store_info = $calculate_data[ 'store_info' ] ?? [];
  61. $store_name = $store_info[ 'store_name' ] ?? '';
  62. $extend = $calculate_data[ 'extend' ] ?? [];
  63. $sell_time = $calculate_data[ 'create_time' ] ?? 0;
  64. if ($sell_time == 0) {
  65. $sell_time = time();
  66. } else {
  67. $sell_time = strtotime($sell_time);
  68. }
  69. $cashier_order_type = $calculate_data[ 'cashier_order_type' ];
  70. $operator = $calculate_data[ 'operator' ] ?? [];
  71. $operator_id = $operator[ 'uid' ] ?? 0;
  72. $operator_name = $operator[ 'username' ] ?? '';
  73. $order_from = 'cashier';
  74. $order_from_name = ( new CashierOrder )->order_from_list[ $order_from ][ 'name' ] ?? '';
  75. $cashier_order_model = new CashierOrder();
  76. //创建订单
  77. $data_order = [
  78. 'order_no' => $order_no,
  79. 'site_id' => $site_id,
  80. 'site_name' => $site_info[ 'name' ],
  81. 'order_name' => $order_name,
  82. 'out_trade_no' => $out_trade_no,
  83. 'member_id' => $member_id,
  84. 'nickname' => $nickname,
  85. 'headimg' => $headimg,
  86. 'mobile' => $mobile,
  87. 'pay_money' => $calculate_data[ 'pay_money' ],
  88. 'goods_money' => $calculate_data[ 'goods_money' ],
  89. 'real_goods_money' => $calculate_data[ 'real_goods_money' ],
  90. 'order_money' => $calculate_data[ 'order_money' ],
  91. 'store_id' => $store_id,
  92. 'create_time' => time(),
  93. 'order_from' => $order_from,
  94. 'order_from_name' => $order_from_name,
  95. 'order_type' => $order_type,
  96. 'order_type_name' => '收银订单',
  97. 'order_status' => 0,
  98. 'order_status_name' => '待支付',
  99. 'order_status_action' => json_encode($cashier_order_model->order_status[ 0 ], JSON_UNESCAPED_UNICODE),
  100. 'cashier_sell_time' => $sell_time,
  101. 'cashier_order_type' => $cashier_order_type,
  102. 'cashier_operator_id' => $operator_id,
  103. 'cashier_operator_name' => $operator_name,
  104. 'order_scene' => 'cashier',
  105. 'goods_num' => $calculate_data[ 'goods_num' ] ?? 1,
  106. 'remark' => $calculate_data[ 'remark' ] ?? ''
  107. ];
  108. $order_id = model('order')->add($data_order);
  109. $calculate_data[ 'order_no' ] = $order_no;
  110. $calculate_data[ 'out_trade_no' ] = $out_trade_no;
  111. $order_goods_id_map = [];
  112. foreach ($goods_list as $goods_k => $goods_info) {
  113. $card_item_id = $goods_info[ 'card_item_id' ] ?? 0;
  114. //订单项目表
  115. $data_order_goods = [
  116. 'order_id' => $order_id,
  117. 'site_id' => $site_id,
  118. 'member_id' => $member_id,
  119. 'goods_id' => $goods_info[ 'goods_id' ],
  120. 'sku_id' => $goods_info[ 'sku_id' ],
  121. 'goods_name' => $goods_info[ 'goods_name' ],
  122. 'sku_name' => $goods_info[ 'sku_name' ],
  123. 'sku_no' => $goods_info[ 'sku_no' ] ?? '',
  124. 'sku_image' => $goods_info[ 'goods_image' ] ?? '',
  125. 'spec_name' => $goods_info[ 'spec_name' ] ?? '',
  126. 'price' => $goods_info[ 'price' ],
  127. 'num' => $goods_info[ 'num' ],
  128. 'goods_money' => $goods_info[ 'goods_money' ],
  129. 'is_virtual' => $goods_info[ 'is_virtual' ] ?? 1,
  130. 'goods_class' => $goods_info[ 'goods_class' ],
  131. 'goods_class_name' => $goods_info[ 'goods_class_name' ],
  132. 'store_id' => $store_id,
  133. 'extend' => json_encode($extend),
  134. 'real_goods_money' => $goods_info[ 'real_goods_money' ],
  135. 'card_item_id' => $card_item_id,
  136. 'card_promotion_money' => $goods_info[ 'card_promotion_money' ] ?? 0,//次卡抵扣优惠
  137. ];
  138. //扣除库存
  139. if (in_array($cashier_order_type, [ 'goods', 'card' ])) {
  140. $stock_result = $this->skuDecStock($goods_info, $store_id);
  141. if ($stock_result[ 'code' ] < 0) {
  142. model('order')->rollback();
  143. return $stock_result;
  144. }
  145. }
  146. $order_goods_id = model('order_goods')->add($data_order_goods);
  147. $goods_list[ $goods_k ][ 'order_goods_id' ] = $order_goods_id;
  148. $order_goods_id_map[ $goods_k ] = $order_goods_id;
  149. }
  150. $calculate_data[ 'order_id' ] = $order_id;
  151. model('order')->commit();
  152. $log_data = array (
  153. 'order_id' => $order_id,
  154. 'action' => 'create',
  155. 'site_id' => $site_id,
  156. 'member_id' => $member_id
  157. );
  158. ( new OrderLog() )->addLog($log_data);
  159. $res = [
  160. 'order_id' => $order_id,
  161. 'out_trade_no' => $out_trade_no
  162. ];
  163. // 生成整体付费支付单据
  164. $pay_model->addPay($site_id, $out_trade_no, $this->pay_type, $order_name, $order_name, $pay_money, '', 'CashierOrderPayNotify', '');
  165. //执行自动关闭
  166. $this->addOrderCronClose($order_id, $site_id); //增加关闭订单自动事件
  167. return $this->success($res);
  168. } catch (\Exception $e) {
  169. model('order')->rollback();
  170. return $this->error('', $e->getMessage() . $e->getFile() . $e->getLine());
  171. }
  172. }
  173. /**
  174. * 计算
  175. * @param $data
  176. * @return mixed
  177. */
  178. public function calculate($data)
  179. {
  180. $data = $this->initStore($data);//初始化仓库门店
  181. $member_id = $data[ 'member_id' ] ?? 0;
  182. $data = $this->initSiteData($data); //初始化站点信息
  183. //商品列表信息
  184. $calculate_result = $this->getItemList($data);
  185. if ($calculate_result[ 'code' ] < 0) {
  186. return $calculate_result;
  187. }
  188. return $calculate_result;
  189. }
  190. /**
  191. * 购买项列表
  192. * @param $data
  193. * @return mixed
  194. */
  195. public function getItemList($data)
  196. {
  197. $type = $data[ 'type' ];
  198. $data_result = event('CashierCalculate', $data, true);
  199. if (empty($data_result)) {
  200. switch ( $type ) {
  201. case 'goods':
  202. $data[ 'cashier_order_type' ] = 'goods';
  203. //处理产品数据
  204. $sku_array = $data[ 'sku_array' ] ?? [];
  205. //消费分为产品和买单(暂)
  206. $product_id_array = [];
  207. $money_array = [];//买单
  208. foreach ($sku_array as $k => $v) {
  209. $money = $v[ 'money' ] ?? 0;
  210. $sku_id = $v[ 'sku_id' ] ?? 0;
  211. $num = $v[ 'num' ] ?? 1;
  212. if ($money > 0) {
  213. $money_array[] = [ 'money' => $money, 'sku_id' => $sku_id, 'num' => $num ];
  214. } else {
  215. if ($sku_id > 0) {
  216. $product_id_array[] = $v;
  217. }
  218. }
  219. }
  220. $data[ 'product_array' ] = $product_id_array;
  221. $calculate_result = $this->toCalculate($data);
  222. if ($calculate_result[ 'code' ] < 0) {
  223. return $calculate_result;
  224. }
  225. $data = $calculate_result[ 'data' ];
  226. $data[ 'money_array' ] = $money_array;
  227. $calculate_result = $this->moneyCalculate($data);
  228. if ($calculate_result[ 'code' ] < 0) {
  229. return $calculate_result;
  230. }
  231. $goods_list = $calculate_result[ 'data' ][ 'goods_list' ] ?? [];
  232. if (empty($goods_list))
  233. return $this->error([], '缺少必填参数商品数据');
  234. $calculate_result[ 'data' ][ 'goods_list' ] = array_reverse($calculate_result[ 'data' ][ 'goods_list' ]);
  235. $data = $calculate_result[ 'data' ];
  236. break;
  237. case 'card'://卡项
  238. $data[ 'cashier_order_type' ] = 'card';
  239. $shop_goods_list_result = $this->cardCalculate($data);
  240. if ($shop_goods_list_result[ 'code' ] < 0) {
  241. return $shop_goods_list_result;
  242. }
  243. $data = $shop_goods_list_result[ 'data' ];
  244. break;
  245. }
  246. } else {
  247. $data = $data_result[ 'data' ];
  248. }
  249. return $this->success($data);
  250. }
  251. /**
  252. * 个性化(主要用于商品)计算
  253. * @param $data
  254. * @return mixed
  255. */
  256. public function toCalculate($data)
  257. {
  258. $product_array = $data[ 'product_array' ] ?? [];
  259. if (!empty($product_array)) {
  260. $shop_goods_list_result = $this->getShopGoodsList($data);
  261. if ($shop_goods_list_result[ 'code' ] < 0) {
  262. return $shop_goods_list_result;
  263. }
  264. $data = $shop_goods_list_result[ 'data' ];
  265. //商品部分的计算
  266. $data = $this->goodsCalculate($data);
  267. //卡项抵消
  268. $delivery_money = $data[ 'delivery_money' ] ?? 0;
  269. $real_goods_money = $data[ 'real_goods_money' ] ?? $data[ 'goods_money' ];
  270. $order_money = moneyFormat($real_goods_money + $delivery_money);
  271. $pay_money = $order_money;
  272. $data[ 'pay_money' ] = $pay_money;
  273. //总结计算
  274. $data[ 'order_money' ] = $order_money;
  275. }
  276. return $this->success($data);
  277. }
  278. /**
  279. * 买单计算
  280. * @param $data
  281. */
  282. public function moneyCalculate($data)
  283. {
  284. $money_array = $data[ 'money_array' ];
  285. if (!empty($money_array)) {
  286. $goods_money = $data[ 'goods_money' ] ?? 0;
  287. $order_name = $data[ 'order_name' ] ?? '';
  288. $order_money = $data[ 'order_money' ] ?? 0;
  289. $pay_money = $data[ 'pay_money' ] ?? 0;
  290. $goods_list = $data[ 'goods_list' ] ?? [];
  291. $goods_image = 'public/uniapp/cashier/cashier-order-money.png';
  292. $goods_num = 0;
  293. foreach ($money_array as $k => $v) {
  294. $num = $v[ 'num' ] ?? 1;
  295. $item_price = $v[ 'money' ];
  296. if ($item_price > 0) {
  297. $sku_id = $v[ 'sku_id' ];
  298. $item_goods_money = moneyFormat($item_price * $num);
  299. $item_order_name = $item_goods_money . '元买单';
  300. $sku_name = '无码商品';
  301. $goods_list[] = array (
  302. 'goods_id' => 0,
  303. 'sku_id' => $sku_id,
  304. 'price' => $item_price,
  305. 'num' => $num,
  306. 'goods_money' => $item_goods_money,
  307. 'goods_name' => $item_order_name,
  308. 'sku_name' => $sku_name,
  309. 'goods_image' => $goods_image,
  310. 'spec_name' => '',
  311. 'goods_class' => 'money',
  312. 'goods_class_name' => '无码商品',
  313. 'real_goods_money' => $item_goods_money
  314. );
  315. $goods_money += $item_goods_money;
  316. $order_money += $order_money;
  317. $pay_money += $pay_money;
  318. $order_name = string_split($order_name, ",", $item_order_name);
  319. $goods_num += $num;
  320. }
  321. }
  322. $data[ 'goods_money' ] = $goods_money;
  323. $data[ 'real_goods_money' ] = $goods_money;
  324. $data[ 'goods_list' ] = $goods_list;
  325. $data[ 'order_name' ] = $order_name;
  326. $data[ 'order_money' ] = $goods_money;
  327. $data[ 'goods_num' ] = $goods_num;
  328. $data[ 'pay_money' ] = $goods_money;
  329. }
  330. return $this->success($data);
  331. }
  332. public function goodsCalculate($data)
  333. {
  334. $site_id = $data[ 'site_id' ];
  335. $goods_list = $data[ 'goods_list' ];
  336. $sku_array = $data[ 'product_array' ];
  337. //具备某个参数(控制是否是单品活动)
  338. //计算订单总额 ,订单总优惠 ,
  339. $goods_money = 0;
  340. $real_goods_money = 0;
  341. foreach ($goods_list as $k => $v) {
  342. // 计算单价 可能情况 (折扣价 会员价)
  343. $goods_item = $v;
  344. $sku_id = $v[ 'sku_id' ];
  345. //商品类主体中应该封装一个函数用于获取商品价格(可能还会有关联会员价 满减 折扣....)
  346. if ($v[ 'is_unify_price' ] == 1) {
  347. $price = moneyFormat($v[ 'sku_price' ]);
  348. } else {
  349. $price = moneyFormat($v[ 'price' ]);
  350. }
  351. $v[ 'price' ] = $price;
  352. $member_price_result = $this->getGoodsMemberPrice($v, $data);
  353. if ($member_price_result[ 'code' ] >= 0) {
  354. $price = $member_price_result[ 'data' ];
  355. $goods_item[ 'is_member_price' ] = true;
  356. }
  357. $num = $goods_item[ 'num' ];//购买数量
  358. $item_goods_money = moneyFormat($price * $num);//商品总额()
  359. $goods_item[ 'price' ] = $price;
  360. $goods_item[ 'goods_money' ] = $item_goods_money;
  361. $goods_item[ 'site_id' ] = $site_id;
  362. // $min_buy = $goods_item[ 'min_buy' ];
  363. // $max_buy = $goods_item[ 'max_buy' ];
  364. // if ($min_buy > 0 && $min_buy > $num)
  365. // $this->setError('GOODS_LESS_MIN_NUM');
  366. //
  367. // if ($max_buy > 0 && $max_buy < $num)
  368. // $this->setError('GOODS_OUT_MAX_NUM');
  369. $item_real_goods_money = $goods_item[ 'goods_money' ];
  370. $goods_item[ 'real_goods_money' ] = $item_real_goods_money;
  371. //卡项
  372. $goods_item = $this->itemCardCalculate($goods_item, $data);
  373. if (empty($goods_item)) {
  374. unset($goods_list[ $k ]);
  375. continue;
  376. } else {
  377. $item_goods_money = $goods_item[ 'goods_money' ];
  378. $item_real_goods_money = $goods_item[ 'real_goods_money' ];
  379. }
  380. $goods_money += $item_goods_money;
  381. $real_goods_money += $item_real_goods_money;
  382. $goods_item[ 'real_goods_money' ] = $item_real_goods_money;
  383. $goods_list[ $k ] = $goods_item;
  384. }
  385. $data[ 'goods_list' ] = $goods_list;
  386. $data[ 'goods_money' ] = $goods_money;
  387. $data[ 'real_goods_money' ] = $real_goods_money;
  388. return $data;
  389. }
  390. /**
  391. * 商品卡项抵扣(抵扣会将商品单价视为0)
  392. * @param $goods_item
  393. * @param $card_item
  394. */
  395. public function itemCardCalculate($goods_item, $data)
  396. {
  397. $card_item_id = $goods_item[ 'card_item_id' ] ?? 0;
  398. $member_id = $data[ 'member_id' ];
  399. $sku_id = $goods_item[ 'sku_id' ];
  400. if ($card_item_id > 0) {
  401. $num = $goods_item[ 'num' ];
  402. $member_card_model = new MemberCard();
  403. $card_item_params = array (
  404. 'member_id' => $member_id,
  405. 'sku_id' => $sku_id,
  406. 'item_id' => $card_item_id
  407. );
  408. $item_card_result = $member_card_model->getUseCardNum($card_item_params);
  409. if ($item_card_result[ 'code' ] < 0) {
  410. return [];
  411. }
  412. $item_card_data = $item_card_result[ 'data' ];
  413. $card_item_info = $item_card_data[ 'card_item_info' ];
  414. $card_info = $item_card_data[ 'card_info' ];
  415. $goods_item[ 'card_item_info' ] = $card_item_info;
  416. $goods_item[ 'card_info' ] = $card_info;
  417. $card_num = $item_card_data[ 'card_num' ];
  418. $goods_item[ 'card_num' ] = $card_num;
  419. if ($card_num > 0) {
  420. if ($num > $card_num) {
  421. $num = $card_num;
  422. }
  423. }
  424. $goods_item[ 'num' ] = $num;
  425. $price = $goods_item[ 'price' ];
  426. $card_promotion_money = moneyFormat($price * $num);
  427. $goods_money = moneyFormat($price * $num);
  428. $goods_item[ 'price' ] = $price;
  429. $goods_item[ 'goods_money' ] = $goods_money;
  430. $real_goods_money = moneyFormat($goods_money - $card_promotion_money);
  431. $real_goods_money = $real_goods_money < 0 ? 0 : $real_goods_money;
  432. $goods_item[ 'card_promotion_money' ] = $card_promotion_money;
  433. $goods_item[ 'real_goods_money' ] = $real_goods_money;
  434. }
  435. return $goods_item;
  436. }
  437. /**
  438. * 获取立即购买商品信息
  439. * @param $data
  440. * @return array
  441. */
  442. public function getShopGoodsList($data)
  443. {
  444. $store_id = $data[ 'store_id' ];
  445. $sku_array = $data[ 'product_array' ];
  446. $sku_ids = array_column($sku_array, 'sku_id');
  447. $sku_data = array_column($sku_array, null, 'sku_id');
  448. $data[ 'sku_data' ] = $sku_data;
  449. $site_id = $data[ 'site_id' ];
  450. $field = 'gs.site_id,gs.goods_id,gs.sku_id,gs.goods_name,gs.sku_name,gs.spec_name,gs.sku_image,g.goods_image,gs.price as sku_price,gs.sku_no,
  451. g.is_virtual,g.min_buy,g.max_buy,g.unit,sgs.stock,
  452. g.goods_class,g.goods_class_name,sgs.price,(sgs.sale_num + g.virtual_sale) as sale_num,g.is_unify_price';
  453. $alias = 'gs';
  454. $join = [
  455. [ 'goods g', 'g.goods_id = gs.goods_id', 'inner' ],
  456. [
  457. 'store_goods_sku sgs',
  458. 'sgs.sku_id = gs.sku_id and sgs.store_id = ' . $store_id,
  459. 'left'
  460. ]
  461. ];
  462. $goods_condition = [
  463. [ 'gs.is_delete', '=', 0 ],
  464. [ 'sgs.status', '=', 1 ],
  465. // [ 'gs.goods_state', '=', 1 ],
  466. [ 'gs.sku_id', 'in', $sku_ids ],
  467. [ 'gs.site_id', '=', $site_id ]
  468. ];
  469. $temp_goods_list = model('goods_sku')->getList($goods_condition, $field, '', $alias, $join);
  470. if (empty($temp_goods_list)) {
  471. return $this->error([], '商品不存在!');
  472. }
  473. foreach ($temp_goods_list as $k => $v) {
  474. $temp_goods_list[ $k ][ 'sale_num' ] = numberFormat($temp_goods_list[ $k ][ 'sale_num' ]);
  475. $temp_goods_list[ $k ][ 'stock' ] = numberFormat($temp_goods_list[ $k ][ 'stock' ]);
  476. }
  477. $column_goods_list = array_column($temp_goods_list, null, 'sku_id');
  478. $order_name = '';
  479. $goods_num = 0;
  480. $goods_list = [];
  481. //分配关联商品数据
  482. foreach ($sku_array as $k => $v) {
  483. $sku_id = $v[ 'sku_id' ];
  484. $goods_info = $column_goods_list[ $sku_id ] ?? 0;
  485. if (empty($goods_info)) {
  486. return $this->error([], '商品不存在!');
  487. }
  488. $order_name = string_split($order_name, ",", $goods_info[ 'goods_name' ]);
  489. $item_num = $v[ 'num' ] ?? 1;
  490. if ($goods_info[ 'goods_class' ] != 6) {
  491. if ($item_num < 1) {
  492. return $this->error([], '商品数量不能小于1!');
  493. }
  494. } else {
  495. if ($item_num < 0) {
  496. return $this->error([], '称重重量不能小于0!');
  497. }
  498. }
  499. $goods_info[ 'card_item_id' ] = $v[ 'card_item_id' ] ?? 0;
  500. $goods_num += $item_num;
  501. $is_virtual = $is_virtual ?? $goods_info[ 'is_virtual' ];
  502. // $goods_info[ 'trade_type' ] = $goods_info['goods_class'];//业务类型
  503. $goods_info[ 'num' ] = $item_num;
  504. $goods_info[ 'sku_image' ] = explode(',', $goods_info[ 'sku_image' ])[ 0 ] ?? '';
  505. $goods_info[ 'goods_image' ] = explode(',', $goods_info[ 'goods_image' ])[ 0 ] ?? '';
  506. $goods_list[ $k ] = $goods_info;
  507. }
  508. $data[ 'goods_list' ] = $goods_list;
  509. $data[ 'order_name' ] = $order_name;
  510. $data[ 'goods_num' ] = $goods_num;
  511. return $this->success($data);
  512. }
  513. /**
  514. * 卡项的计算
  515. * @param $data
  516. */
  517. public function cardCalculate($data)
  518. {
  519. $store_id = $data[ 'store_id' ];
  520. $sku_array = $data[ 'sku_array' ];
  521. $sku_ids = array_column($sku_array, 'sku_id');
  522. $sku_num_array = array_column($sku_array, 'num', 'sku_id');
  523. // $sku_ids = $data['sku_ids'];//sku_id数组weight
  524. $site_id = $data[ 'site_id' ];
  525. $field = 'gs.site_id,gs.goods_id,gs.sku_id,gs.goods_name,gs.sku_name,gs.spec_name,gs.sku_image,g.goods_image,sgs.price,gs.price as sku_price,gs.sku_no,
  526. g.is_virtual,g.unit,sgs.stock,
  527. g.min_buy,g.max_buy,
  528. g.goods_class,g.goods_class_name,(sgs.sale_num + g.virtual_sale) as sale_num,g.is_unify_price';
  529. $alias = 'gs';
  530. $join = [
  531. [ 'goods g', 'g.goods_id = gs.goods_id', 'inner' ],
  532. [
  533. 'store_goods_sku sgs',
  534. 'sgs.sku_id = gs.sku_id and sgs.store_id = ' . $store_id,
  535. 'left'
  536. ]
  537. ];
  538. $goods_condition = [
  539. [ 'gs.is_delete', '=', 0 ],
  540. [ 'sgs.status', '=', 1 ],
  541. [ 'gs.sku_id', 'in', $sku_ids ],
  542. [ 'gs.site_id', '=', $site_id ],
  543. ];
  544. $goods_list = model('goods_sku')->getList($goods_condition, $field, '', $alias, $join);
  545. $order_name = '';
  546. $goods_num = 0;
  547. foreach ($goods_list as $k => $v) {
  548. $sku_id = $v[ 'sku_id' ];
  549. $order_name = string_split($order_name, ",", $v[ 'goods_name' ]);
  550. $item_num = $sku_num_array[ $sku_id ] ?? 1;
  551. $goods_num += $item_num;
  552. $goods_info = $v;
  553. $is_virtual = $is_virtual ?? $v[ 'is_virtual' ];
  554. $goods_info[ 'num' ] = $item_num;
  555. $goods_info[ 'sale_num' ] = numberFormat($goods_info[ 'sale_num' ]);
  556. $goods_info[ 'stock' ] = numberFormat($goods_info[ 'stock' ]);
  557. $goods_list[ $k ] = $goods_info;
  558. }
  559. $data[ 'goods_list' ] = $goods_list;
  560. $data[ 'order_name' ] = $order_name;
  561. $data[ 'goods_num' ] = $goods_num;
  562. $goods_list = $data[ 'goods_list' ];
  563. //具备某个参数(控制是否是单品活动)
  564. //计算订单总额 ,订单总优惠 ,
  565. $goods_money = 0;
  566. $real_goods_money = 0;
  567. foreach ($goods_list as $k => $v) {
  568. // 计算单价 可能情况 (折扣价 会员价)
  569. $goods_item = $v;
  570. //商品类主体中应该封装一个函数用于获取商品价格(可能还会有关联会员价 满减 折扣....)
  571. if ($v[ 'is_unify_price' ] == 1) {
  572. $price = moneyFormat($v[ 'sku_price' ]);
  573. } else {
  574. $price = moneyFormat($v[ 'price' ]);
  575. }
  576. $v[ 'price' ] = $price;
  577. $member_price_result = $this->getGoodsMemberPrice($v, $data);
  578. if ($member_price_result[ 'code' ] >= 0) {
  579. $price = $member_price_result[ 'data' ];
  580. $goods_item[ 'is_member_price' ] = true;
  581. }
  582. $goods_item[ 'price' ] = $price;
  583. $num = $goods_item[ 'num' ];//购买数量
  584. $item_goods_money = moneyFormat($price * $num);//商品总额()
  585. // $min_buy = $goods_item[ 'min_buy' ];
  586. // $max_buy = $goods_item[ 'max_buy' ];
  587. // if ($min_buy > 0 && $min_buy > $num) {
  588. // $this->setError('GOODS_LESS_MIN_NUM');
  589. // }
  590. // if ($max_buy > 0 && $max_buy < $num) {
  591. // $this->setError('GOODS_OUT_MAX_NUM');
  592. // }
  593. $goods_money += $item_goods_money;
  594. $item_real_goods_money = $item_goods_money;
  595. $real_goods_money += $item_real_goods_money;
  596. $goods_item[ 'goods_money' ] = $item_goods_money;
  597. $goods_item[ 'real_goods_money' ] = $item_real_goods_money;
  598. $goods_item[ 'sku_image' ] = explode(',', $goods_item[ 'sku_image' ])[ 0 ] ?? '';
  599. $goods_item[ 'goods_image' ] = explode(',', $goods_item[ 'goods_image' ])[ 0 ] ?? '';
  600. $goods_list[ $k ] = $goods_item;
  601. }
  602. $data[ 'goods_list' ] = $goods_list;
  603. $data[ 'goods_money' ] = moneyFormat($goods_money);
  604. $data[ 'real_goods_money' ] = moneyFormat($real_goods_money);
  605. $delivery_money = $data[ 'delivery_money' ] ?? 0;
  606. $order_money = moneyFormat($real_goods_money + $delivery_money);
  607. $pay_money = $order_money;
  608. //总结计算
  609. $data[ 'order_money' ] = $order_money;
  610. $data[ 'pay_money' ] = $pay_money;
  611. return $this->success($data);
  612. }
  613. /**
  614. * 设置错误
  615. * @param $error_code
  616. */
  617. protected function setError($error_code, $error_printf = [])
  618. {
  619. $this->error = $error_code;
  620. $this->error_printf = $error_printf;
  621. }
  622. /**
  623. * 校验错误
  624. * @return bool
  625. */
  626. public function checkError()
  627. {
  628. if (!empty($this->error)) {
  629. $error_msg = $this->getError();
  630. return $this->error([ 'error_code' => $this->error, 'error_msg' => $error_msg ], $error_msg);
  631. } else {
  632. return $this->success();
  633. }
  634. }
  635. /**
  636. * 获取错误
  637. * @return mixed
  638. */
  639. protected function getError()
  640. {
  641. $error_list = [
  642. 'GOODS_STOCK_EMPTY' => '%s库存不足%s',
  643. 'ADDRESS_EMPTY' => '%s收货地址必须选择!%s',
  644. 'TRADE_TYPE_EMPTY' => '%s配送方式必须选择!%s',
  645. 'GOODS_LESS_MIN_NUM' => '%s商品数量不能小于最小购买量%s',
  646. 'GOODS_OUT_MAX_NUM' => '%s商品数量不能超出最大购买量%s',
  647. 'MOBILE_EMPTY' => '%s联系人手机号不能为空%s',
  648. 'NAME_EMPTY' => '%s联系人名称不能为空%s',
  649. 'YUEYUE_ERROR' => '%s %s',
  650. ];
  651. $error_msg = $error_list[ $this->error ] ?? '';
  652. return sprintf($error_msg, $this->error_printf[ 0 ] ?? '', $this->error_printf[ 1 ] ?? '');
  653. }
  654. }