CashierOrderCreate.php 29 KB

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