buy_1.logic.php 57 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255
  1. <?php
  2. /**
  3. * 购买行为
  4. *
  5. * by 33hao.com 好商城V3 运营版
  6. */
  7. defined('InShopNC') or exit('Access Invalid!');
  8. require_once (BASE_ROOT_PATH . '/helper/activity_helper.php');
  9. class buy_1Logic
  10. {
  11. /**
  12. * 取得商品最新的属性及促销[购物车]
  13. * @param unknown $cart_list
  14. */
  15. public function getGoodsCartList($cart_list)
  16. {
  17. $cart_list = $this->_getOnlineCartList($cart_list);
  18. //优惠套装
  19. $this->_getBundlingCartList($cart_list);
  20. //抢购
  21. $this->getGroupbuyCartList($cart_list);
  22. //限时折扣
  23. $this->getXianshiCartList($cart_list);
  24. //赠品
  25. $this->_getGiftCartList($cart_list);
  26. return $cart_list;
  27. }
  28. /**
  29. * 取得商品最新的属性及促销[立即购买]
  30. * @param int $goods_id
  31. * @param int $quantity
  32. * @return array
  33. */
  34. public function getGoodsOnlineInfo($goods_id,$quantity)
  35. {
  36. $goods_info = $this->_getGoodsOnlineInfo($goods_id,$quantity);
  37. //抢购
  38. $this->getGroupbuyInfo($goods_info);
  39. //限时折扣
  40. $this->getXianshiInfo($goods_info,$goods_info['goods_num']);
  41. //赠品
  42. $this->_getGoodsGiftList($goods_info);
  43. return $goods_info;
  44. }
  45. /**
  46. * 商品金额计算(分别对每个商品/优惠套装小计、每个店铺小计)
  47. * @param unknown $store_cart_list 以店铺ID分组的购物车商品信息
  48. * @return array
  49. */
  50. public function calcCartList($store_cart_list) {
  51. if (empty($store_cart_list) || !is_array($store_cart_list)) return array($store_cart_list,array(),0);
  52. //存放每个店铺的商品总金额
  53. $store_goods_total = array();
  54. //存放本次下单所有店铺商品总金额
  55. $order_goods_total = 0;
  56. foreach ($store_cart_list as $store_id => $store_cart) {
  57. $tmp_amount = 0;
  58. foreach ($store_cart as $key => $cart_info) {
  59. $store_cart[$key]['goods_total'] = ncPriceFormat($cart_info['goods_price'] * $cart_info['goods_num']);
  60. $store_cart[$key]['goods_image_url'] = cthumb($store_cart[$key]['goods_image']);
  61. $tmp_amount += $store_cart[$key]['goods_total'];
  62. }
  63. $store_cart_list[$store_id] = $store_cart;
  64. $store_goods_total[$store_id] = ncPriceFormat($tmp_amount);
  65. }
  66. return array($store_cart_list,$store_goods_total);
  67. }
  68. /**
  69. * 取得店铺级优惠 - 跟据商品金额返回每个店铺当前符合的一条活动规则,如果有赠品,则自动追加到购买列表,价格为0
  70. * @param unknown $store_goods_total 每个店铺的商品金额小计,以店铺ID为下标
  71. * @return array($premiums_list,$mansong_rule_list) 分别为赠品列表[下标自增],店铺满送规则列表[店铺ID为下标]
  72. */
  73. public function getMansongRuleCartListByTotal($store_goods_total) {
  74. if (!C('promotion_allow') || empty($store_goods_total) || !is_array($store_goods_total)) return array(array(),array());
  75. $model_mansong = Model('p_mansong');
  76. $model_goods = Model('goods');
  77. //定义赠品数组,下标为店铺ID
  78. $premiums_list = array();
  79. //定义满送活动数组,下标为店铺ID
  80. $mansong_rule_list = array();
  81. foreach ($store_goods_total as $store_id => $goods_total) {
  82. $rule_info = $model_mansong->getMansongRuleByStoreID($store_id,$goods_total);
  83. if (is_array($rule_info) && !empty($rule_info)) {
  84. //即不减金额,也找不到促销商品时(已下架),此规则无效
  85. if (empty($rule_info['discount']) && empty($rule_info['mansong_goods_name'])) {
  86. continue;
  87. }
  88. $rule_info['desc'] = $this->_parseMansongRuleDesc($rule_info);
  89. $rule_info['discount'] = ncPriceFormat($rule_info['discount']);
  90. $mansong_rule_list[$store_id] = $rule_info;
  91. //如果赠品在售,有库存,则追加到购买列表
  92. if (!empty($rule_info['mansong_goods_name']) && !empty($rule_info['goods_storage'])) {
  93. $data = array();
  94. $data['goods_id'] = $rule_info['goods_id'];
  95. $data['goods_name'] = $rule_info['mansong_goods_name'];
  96. $data['goods_num'] = 1;
  97. $data['goods_price'] = 0.00;
  98. $data['goods_image'] = $rule_info['goods_image'];
  99. $data['goods_image_url'] = cthumb($rule_info['goods_image']);
  100. $data['goods_storage'] = $rule_info['goods_storage'];
  101. $premiums_list[$store_id][] = $data;
  102. }
  103. }
  104. }
  105. return array($premiums_list,$mansong_rule_list);
  106. }
  107. public function getOptionalGoods($store_cart_list)
  108. {
  109. $result = [];
  110. foreach ($store_cart_list as $store_id => $cart_list)
  111. {
  112. if($store_id == 6)
  113. {
  114. $matcher = new optional_match($cart_list);
  115. $op_result = $matcher->match();
  116. if($op_result != false) {
  117. $result[$store_id] = $op_result;
  118. } else {
  119. $result[$store_id] = array('discount' => 0,'goods_nums' => []);
  120. }
  121. }
  122. else {
  123. $result[$store_id] = array('discount' => 0,'goods_nums' => []);
  124. }
  125. }
  126. return $result;
  127. }
  128. /**
  129. * 重新计算每个店铺最终商品总金额(最初计算金额减去各种优惠/加运费)
  130. * @param array $store_goods_total 店铺商品总金额
  131. * @param array $preferential_array 店铺优惠活动内容
  132. * @param string $preferential_type 优惠类型,目前只有一个 'mansong'
  133. * @return array 返回扣除优惠后的店铺商品总金额
  134. */
  135. public function reCalcGoodsTotal($store_goods_total, $preferential_array, $preferential_type) {
  136. $deny = empty($store_goods_total) || !is_array($store_goods_total) || empty($preferential_array) || !is_array($preferential_array);
  137. if ($deny) return $store_goods_total;
  138. switch ($preferential_type)
  139. {
  140. case 'mansong':
  141. if (!C('promotion_allow')) return $store_goods_total;
  142. foreach ($preferential_array as $store_id => $rule_info) {
  143. if (is_array($rule_info) && $rule_info['discount'] > 0) {
  144. $store_goods_total[$store_id] -= $rule_info['discount'];
  145. }
  146. }
  147. break;
  148. case 'voucher':
  149. if (!C('voucher_allow')) return $store_goods_total;
  150. foreach ($preferential_array as $store_id => $voucher_info) {
  151. $store_goods_total[$store_id] -= $voucher_info['voucher_price'];
  152. }
  153. break;
  154. case 'freight':
  155. foreach ($preferential_array as $store_id => $freight_total) {
  156. $store_goods_total[$store_id] += $freight_total;
  157. }
  158. break;
  159. case 'optional_goods':
  160. foreach ($preferential_array as $store_id => $value) {
  161. $store_goods_total[$store_id] -= $value['discount'];
  162. }
  163. break;
  164. }
  165. return $store_goods_total;
  166. }
  167. /**
  168. * 取得店铺可用的代金券
  169. * @param array $store_goods_total array(店铺ID=>商品总金额)
  170. * @return array
  171. */
  172. public function getStoreAvailableVoucherList($store_goods_total, $member_id) {
  173. if (!C('voucher_allow')) return $store_goods_total;
  174. $voucher_list = array();
  175. $model_voucher = Model('voucher');
  176. foreach ($store_goods_total as $store_id => $goods_total) {
  177. $condition = array();
  178. $condition['voucher_store_id'] = $store_id;
  179. $condition['voucher_owner_id'] = $member_id;
  180. $voucher_list[$store_id] = $model_voucher->getCurrentAvailableVoucher($condition,$goods_total);
  181. }
  182. return $voucher_list;
  183. }
  184. /**
  185. * 验证传过来的代金券是否可用有效,如果无效,直接删除
  186. * @param array $input_voucher_list 代金券列表
  187. * @param array $store_goods_total (店铺ID=>商品总金额)
  188. * @return array
  189. */
  190. public function reParseVoucherList($input_voucher_list = array(), $store_goods_total = array(), $member_id) {
  191. if (empty($input_voucher_list) || !is_array($input_voucher_list)) return array();
  192. $store_voucher_list = $this->getStoreAvailableVoucherList($store_goods_total, $member_id);
  193. foreach ($input_voucher_list as $store_id => $voucher) {
  194. $tmp = $store_voucher_list[$store_id];
  195. if (is_array($tmp) && isset($tmp[$voucher['voucher_t_id']])) {
  196. $input_voucher_list[$store_id]['voucher_id'] = $tmp[$voucher['voucher_t_id']]['voucher_id'];
  197. $input_voucher_list[$store_id]['voucher_code'] = $tmp[$voucher['voucher_t_id']]['voucher_code'];
  198. $input_voucher_list[$store_id]['voucher_owner_id'] = $tmp[$voucher['voucher_t_id']]['voucher_owner_id'];
  199. } else {
  200. unset($input_voucher_list[$store_id]);
  201. }
  202. }
  203. return $input_voucher_list;
  204. }
  205. /**
  206. * 判断商品是不是限时折扣中,如果购买数量若>=规定的下限,按折扣价格计算,否则按原价计算
  207. * @param array $goods_info
  208. * @param number $quantity 购买数量
  209. */
  210. public function getXianshiInfo( & $goods_info, $quantity)
  211. {
  212. if (empty($quantity)) $quantity = 1;
  213. if (!C('promotion_allow') || empty($goods_info['xianshi_info'])) return ;
  214. $goods_info['xianshi_info']['down_price'] = ncPriceFormat($goods_info['goods_price'] - $goods_info['xianshi_info']['xianshi_price']);
  215. if ($quantity >= $goods_info['xianshi_info']['lower_limit']) {
  216. $goods_info['goods_price'] = $goods_info['xianshi_info']['xianshi_price'];
  217. $goods_info['promotions_id'] = $goods_info['xianshi_info']['xianshi_id'];
  218. $goods_info['ifxianshi'] = true;
  219. }
  220. }
  221. /**
  222. * 输出有货到付款时,在线支付和货到付款及每种支付下商品数量和详细列表
  223. * @param $buy_list 商品列表
  224. * @return 返回 以支付方式为下标分组的商品列表
  225. */
  226. public function getOfflineGoodsPay($buy_list) {
  227. //以支付方式为下标,存放购买商品
  228. $buy_goods_list = array();
  229. $offline_pay = Model('payment')->getPaymentOpenInfo(array('payment_code'=>'offline'));
  230. if ($offline_pay) {
  231. //下单里包括平台自营商品并且平台已开启货到付款,则显示货到付款项及对应商品数量,取出支持货到付款的店铺ID组成的数组,目前就一个,DEFAULT_PLATFORM_STORE_ID
  232. $offline_store_id_array = model('store')->getOwnShopIds();
  233. foreach ($buy_list as $value) {
  234. //if (in_array($value['store_id'],$offline_store_id_array)) {
  235. $buy_goods_list['offline'][] = $value;
  236. //} else {
  237. // $buy_goods_list['online'][] = $value;
  238. //}
  239. }
  240. }
  241. return $buy_goods_list;
  242. }
  243. /**
  244. * 计算每个店铺(所有店铺级优惠活动)总共优惠多少金额
  245. * @param array $store_goods_total 最初店铺商品总金额
  246. * @param array $store_final_goods_total 去除各种店铺级促销后,最终店铺商品总金额(不含运费)
  247. * @return array
  248. */
  249. public function getStorePromotionTotal($store_goods_total, $store_final_goods_total) {
  250. if (!is_array($store_goods_total) || !is_array($store_final_goods_total)) return array();
  251. $store_promotion_total = array();
  252. foreach ($store_goods_total as $store_id => $goods_total) {
  253. $store_promotion_total[$store_id] = abs($goods_total - $store_final_goods_total[$store_id]);
  254. }
  255. return $store_promotion_total;
  256. }
  257. /**
  258. * 返回需要计算运费的店铺ID组成的数组 和 免运费店铺ID及免运费下限金额描述
  259. * @param array $store_goods_total 每个店铺的商品金额小计,以店铺ID为下标
  260. * @return array
  261. */
  262. public function getStoreFreightDescList($store_goods_total)
  263. {
  264. if (empty($store_goods_total) || !is_array($store_goods_total)) return array(array(),array());
  265. //定义返回数组
  266. $need_calc_sid_array = array();
  267. $cancel_calc_sid_array = array();
  268. //如果商品金额未达到免运费设置下线,则需要计算运费
  269. $condition = array('store_id' => array('in',array_keys($store_goods_total)));
  270. $store_list = Model('store')->getStoreOnlineList($condition,null,'','store_id,store_free_price');
  271. foreach ($store_list as $store_info) {
  272. $limit_price = floatval($store_info['store_free_price']);
  273. if ($limit_price == 0 || $limit_price > $store_goods_total[$store_info['store_id']]) {
  274. //需要计算运费
  275. $need_calc_sid_array[] = $store_info['store_id'];
  276. $need_calc_sid_array['desc'] = sprintf('满%s免运费',$limit_price); // add by liax
  277. } else {
  278. //返回免运费金额下限
  279. $cancel_calc_sid_array[$store_info['store_id']]['free_price'] = $limit_price;
  280. $cancel_calc_sid_array[$store_info['store_id']]['desc'] = sprintf('满%s免运费',$limit_price);
  281. }
  282. }
  283. return array($need_calc_sid_array,$cancel_calc_sid_array);
  284. }
  285. /**
  286. * 取得店铺运费(使用运费模板的商品运费不会计算,但会返回模板信息)
  287. * 先将免运费的店铺运费置0,然后算出店铺里没使用运费模板的商品运费之和 ,存到iscalced下标中
  288. * 然后再计算使用运费模板的信息(array(店铺ID=>array(运费模板ID=>购买数量)),放到nocalced下标里
  289. * @param array $buy_list 购买商品列表
  290. * @param array $free_freight_sid_list 免运费的店铺ID数组
  291. */
  292. public function getStoreFreightList($buy_list = array(), $free_freight_sid_list) {
  293. //定义返回数组
  294. $return = array();
  295. //先将免运费的店铺运费置0(格式:店铺ID=>0)
  296. $freight_list = array();
  297. if (!empty($free_freight_sid_list) && is_array($free_freight_sid_list)) {
  298. foreach ($free_freight_sid_list as $store_id) {
  299. $freight_list[$store_id] = 0;
  300. }
  301. }
  302. //然后算出店铺里没使用运费模板(优惠套装商品除外)的商品运费之和(格式:店铺ID=>运费)
  303. //定义数组,存放店铺优惠套装商品运费总额 store_id=>运费
  304. $store_bl_goods_freight = array();
  305. foreach ($buy_list as $key => $goods_info) {
  306. //免运费店铺的商品不需要计算
  307. if (in_array($goods_info['store_id'], $free_freight_sid_list)) {
  308. unset($buy_list[$key]);
  309. continue;
  310. }
  311. //优惠套装商品运费另算
  312. if (intval($goods_info['bl_id'])) {
  313. unset($buy_list[$key]);
  314. $store_bl_goods_freight[$goods_info['store_id']] = $goods_info['bl_id'];
  315. continue;
  316. }
  317. if (!intval($goods_info['transport_id']) && !in_array($goods_info['store_id'],$free_freight_sid_list)) {
  318. $freight_list[$goods_info['store_id']] += $goods_info['goods_freight'];
  319. unset($buy_list[$key]);
  320. }
  321. }
  322. //计算优惠套装商品运费
  323. if (!empty($store_bl_goods_freight)) {
  324. $model_bl = Model('p_bundling');
  325. foreach (array_unique($store_bl_goods_freight) as $store_id => $bl_id) {
  326. $bl_info = $model_bl->getBundlingInfo(array('bl_id'=>$bl_id));
  327. if (!empty($bl_info)) {
  328. $freight_list[$store_id] += $bl_info['bl_freight'];
  329. }
  330. }
  331. }
  332. $return['iscalced'] = $freight_list;
  333. //最后再计算使用运费模板的信息(店铺ID,运费模板ID,购买数量),使用使用相同运费模板的商品数量累加
  334. $freight_list = array();
  335. foreach ($buy_list as $goods_info) {
  336. $freight_list[$goods_info['store_id']][$goods_info['transport_id']] += $goods_info['goods_num'];
  337. }
  338. $return['nocalced'] = $freight_list;
  339. return $return;
  340. }
  341. /**
  342. * 根据地区选择计算出所有店铺最终运费
  343. * @param array $freight_list 运费信息(店铺ID,运费,运费模板ID,购买数量)
  344. * @param int $city_id 市级ID
  345. * @return array 返回店铺ID=>运费
  346. */
  347. public function calcStoreFreight($freight_list, $city_id) {
  348. if (!is_array($freight_list) || empty($freight_list) || empty($city_id)) return;
  349. //免费和固定运费计算结果
  350. $return_list = $freight_list['iscalced'];
  351. //使用运费模板的信息(array(店铺ID=>array(运费模板ID=>购买数量))
  352. $nocalced_list = $freight_list['nocalced'];
  353. //然后计算使用运费运费模板的在该$city_id时的运费值
  354. if (!empty($nocalced_list) && is_array($nocalced_list)) {
  355. //如果有商品使用的运费模板,先计算这些商品的运费总金额
  356. $model_transport = Model('transport');
  357. foreach ($nocalced_list as $store_id => $value) {
  358. if (is_array($value)) {
  359. foreach ($value as $transport_id => $buy_num) {
  360. $freight_total = $model_transport->calc_transport($transport_id,$buy_num, $city_id);
  361. if (empty($return_list[$store_id])) {
  362. $return_list[$store_id] = $freight_total;
  363. } else {
  364. $return_list[$store_id] += $freight_total;
  365. }
  366. }
  367. }
  368. }
  369. }
  370. return $return_list;
  371. }
  372. /**
  373. * 追加赠品到下单列表,并更新购买数量
  374. * @param array $store_cart_list 购买列表
  375. * @param array $store_premiums_list 赠品列表
  376. * @param array $store_mansong_rule_list 满即送规则
  377. */
  378. public function appendPremiumsToCartList($store_cart_list, $store_premiums_list = array(), $store_mansong_rule_list = array(), $member_id) {
  379. if (empty($store_cart_list)) return array();
  380. //处理商品级赠品
  381. foreach ($store_cart_list as $store_id => $cart_list) {
  382. foreach ($cart_list as $cart_info) {
  383. if (empty($cart_info['gift_list'])) continue;
  384. if (!is_array($store_premiums_list)) $store_premiums_list = array();
  385. if (!array_key_exists($store_id,$store_premiums_list)) $store_premiums_list[$store_id] = array();
  386. $zenpin_info = array();
  387. foreach ($cart_info['gift_list'] as $gift_info) {
  388. $zenpin_info['goods_id'] = $gift_info['gift_goodsid'];
  389. $zenpin_info['goods_name'] = $gift_info['gift_goodsname'];
  390. $zenpin_info['goods_image'] = $gift_info['gift_goodsimage'];
  391. $zenpin_info['goods_storage'] = $gift_info['goods_storage'];
  392. $zenpin_info['goods_num'] = $cart_info['goods_num'] * $gift_info['gift_amount'];
  393. $store_premiums_list[$store_id][] = $zenpin_info;
  394. }
  395. }
  396. }
  397. //取得每种商品的库存[含赠品]
  398. $goods_storage_quantity = $this->_getEachGoodsStorageQuantity($store_cart_list,$store_premiums_list);
  399. //取得每种商品的购买量[不含赠品]
  400. $goods_buy_quantity = $this->_getEachGoodsBuyQuantity($store_cart_list);
  401. foreach ($goods_buy_quantity as $goods_id => $quantity) {
  402. $goods_storage_quantity[$goods_id] -= $quantity;
  403. if ($goods_storage_quantity[$goods_id] < 0) {
  404. //商品库存不足,请重购买
  405. return false;
  406. }
  407. }
  408. //将赠品追加到购买列表
  409. if(is_array($store_premiums_list)) {
  410. foreach ($store_premiums_list as $store_id => $goods_list) {
  411. $zp_list = array();
  412. $gift_desc = '';
  413. foreach ($goods_list as $goods_info) {
  414. //如果没有库存了,则不再送赠品
  415. if ($goods_storage_quantity[$goods_info['goods_id']] == 0) {
  416. $gift_desc = ',赠品库存不足,未能全部送出 ';
  417. continue;
  418. }
  419. $new_data = array();
  420. $new_data['buyer_id'] = $member_id;
  421. $new_data['store_id'] = $store_id;
  422. $new_data['store_name'] = $store_cart_list[$store_id][0]['store_name'];
  423. $new_data['goods_id'] = $goods_info['goods_id'];
  424. $new_data['goods_name'] = $goods_info['goods_name'];
  425. $new_data['goods_price'] = 0;
  426. $new_data['goods_image'] = $goods_info['goods_image'];
  427. $new_data['bl_id'] = 0;
  428. $new_data['state'] = true;
  429. $new_data['storage_state'] = true;
  430. $new_data['gc_id'] = 0;
  431. $new_data['transport_id'] = 0;
  432. $new_data['goods_freight'] = 0;
  433. $new_data['goods_vat'] = 0;
  434. $new_data['goods_total'] = 0;
  435. $new_data['ifzengpin'] = true;
  436. //计算赠送数量,有就赠,赠完为止
  437. if ($goods_storage_quantity[$goods_info['goods_id']] - $goods_info['goods_num'] >= 0) {
  438. $goods_buy_quantity[$goods_info['goods_id']] += $goods_info['goods_num'];
  439. $goods_storage_quantity[$goods_info['goods_id']] -= $goods_info['goods_num'];
  440. $new_data['goods_num'] = $goods_info['goods_num'];
  441. } else {
  442. $new_data['goods_num'] = $goods_storage_quantity[$goods_info['goods_id']];
  443. $goods_buy_quantity[$goods_info['goods_id']] += $goods_storage_quantity[$goods_info['goods_id']];
  444. $goods_storage_quantity[$goods_info['goods_id']] = 0;
  445. }
  446. if (array_key_exists($goods_info['goods_id'],$zp_list)) {
  447. $zp_list[$goods_info['goods_id']]['goods_num'] += $new_data['goods_num'];
  448. } else {
  449. $zp_list[$goods_info['goods_id']] = $new_data;
  450. }
  451. }
  452. sort($zp_list);
  453. $store_cart_list[$store_id] = array_merge($store_cart_list[$store_id],$zp_list);
  454. $store_mansong_rule_list[$store_id]['desc'] .= $gift_desc;
  455. $store_mansong_rule_list[$store_id]['desc'] = trim($store_mansong_rule_list[$store_id]['desc'],',');
  456. }
  457. }
  458. return array($store_cart_list,$goods_buy_quantity,$store_mansong_rule_list);
  459. }
  460. /**
  461. * 充值卡支付,依次循环每个订单
  462. * 如果充值卡足够就单独支付了该订单,如果不足就暂时冻结,等API支付成功了再彻底扣除
  463. */
  464. public function rcbPay($order_list, $input, $buyer_info) {
  465. $member_id = $buyer_info['member_id'];
  466. $member_name = $buyer_info['member_name'];
  467. $available_rcb_amount = floatval($buyer_info['available_rc_balance']);
  468. if ($available_rcb_amount <= 0) return;
  469. $model_order = Model('order');
  470. $model_pd = Model('predeposit');
  471. foreach ($order_list as $key => $order_info) {
  472. //货到付款的订单跳过
  473. if ($order_info['payment_code'] == 'offline') continue;
  474. $order_amount = floatval($order_info['order_amount']);
  475. $data_pd = array();
  476. $data_pd['member_id'] = $member_id;
  477. $data_pd['member_name'] = $member_name;
  478. $data_pd['amount'] = $order_info['order_amount'];
  479. $data_pd['order_sn'] = $order_info['order_sn'];
  480. if ($available_rcb_amount >= $order_amount) {
  481. //立即支付,订单支付完成
  482. $model_pd->changeRcb('order_pay',$data_pd);
  483. $available_rcb_amount -= $order_amount;
  484. //记录订单日志(已付款)
  485. $data = array();
  486. $data['order_id'] = $order_info['order_id'];
  487. $data['log_role'] = 'buyer';
  488. $data['log_msg'] = L('order_log_pay');
  489. $data['log_orderstate'] = ORDER_STATE_PAY;
  490. $insert = $model_order->addOrderLog($data);
  491. if (!$insert) {
  492. throw new Exception('记录订单充值卡支付日志出现错误');
  493. }
  494. //订单状态 置为已支付
  495. $data_order = array();
  496. $order_list[$key]['order_state'] = $data_order['order_state'] = ORDER_STATE_PAY;
  497. $data_order['payment_time'] = time();
  498. $data_order['payment_code'] = 'predeposit';
  499. $data_order['rcb_amount'] = $order_amount;
  500. $result = $model_order->editOrder($data_order,array('order_id'=>$order_info['order_id']));
  501. if (!$result) {
  502. throw new Exception('订单更新失败');
  503. }
  504. // 发送商家提醒
  505. $param = array();
  506. $param['code'] = 'new_order';
  507. $param['store_id'] = $order_info['store_id'];
  508. $param['param'] = array(
  509. 'order_sn' => $order_info['order_sn']
  510. );
  511. QueueClient::push('sendStoreMsg', $param);
  512. } else {
  513. //暂冻结充值卡,后面还需要 API彻底完成支付
  514. if ($available_rcb_amount > 0) {
  515. $data_pd['amount'] = $available_rcb_amount;
  516. $model_pd->changeRcb('order_freeze',$data_pd);
  517. //支付金额保存到订单
  518. $data_order = array();
  519. $order_list[$key]['rcb_amount'] = $data_order['rcb_amount'] = $available_rcb_amount;
  520. $result = $model_order->editOrder($data_order,array('order_id'=>$order_info['order_id']));
  521. $available_rcb_amount = 0;
  522. if (!$result) {
  523. throw new Exception('订单更新失败');
  524. }
  525. }
  526. }
  527. }
  528. return $order_list;
  529. }
  530. /**
  531. * 预存款支付,依次循环每个订单
  532. * 如果预存款足够就单独支付了该订单,如果不足就暂时冻结,等API支付成功了再彻底扣除
  533. */
  534. private function bonus_amount($order_info)
  535. {
  536. $bonus_amount = unserialize($order_info['bonus_amount']);
  537. $amount = floatval($bonus_amount['user_bonus']) + floatval($bonus_amount['room_bonus']);
  538. return $amount;
  539. }
  540. public function pdPay($order_list, $input, $buyer_info)
  541. {
  542. $member_id = $buyer_info['member_id'];
  543. $member_name = $buyer_info['member_name'];
  544. $available_pd_amount = floatval($buyer_info['available_predeposit']);
  545. if ($available_pd_amount <= 0) return;
  546. $model_order = Model('order');
  547. $model_pd = Model('predeposit');
  548. foreach ($order_list as $order_info)
  549. {
  550. //货到付款的订单、已经充值卡支付的订单跳过
  551. if ($order_info['payment_code'] == 'offline') continue;
  552. if ($order_info['order_state'] == ORDER_STATE_PAY) continue;
  553. $order_amount = floatval($order_info['order_amount']) - floatval($order_info['rcb_amount']) - $this->bonus_amount($order_info);
  554. $data_pd = array();
  555. $data_pd['member_id'] = $member_id;
  556. $data_pd['member_name'] = $member_name;
  557. $data_pd['amount'] = $order_amount;
  558. $data_pd['order_sn'] = $order_info['order_sn'];
  559. if ($available_pd_amount >= $order_amount)
  560. {
  561. //预存款立即支付,订单支付完成
  562. $model_pd->changePd('order_pay',$data_pd);
  563. $available_pd_amount -= $order_amount;
  564. //支付被冻结的充值卡
  565. $rcb_amount = floatval($order_info['rcb_amount']);
  566. if ($rcb_amount > 0) {
  567. $data_pd = array();
  568. $data_pd['member_id'] = $member_id;
  569. $data_pd['member_name'] = $member_name;
  570. $data_pd['amount'] = $rcb_amount;
  571. $data_pd['order_sn'] = $order_info['order_sn'];
  572. $model_pd->changeRcb('order_comb_pay',$data_pd);
  573. }
  574. //记录订单日志(已付款)
  575. $data = array();
  576. $data['order_id'] = $order_info['order_id'];
  577. $data['log_role'] = 'buyer';
  578. $data['log_msg'] = L('order_log_pay');
  579. $data['log_orderstate'] = ORDER_STATE_PAY;
  580. $insert = $model_order->addOrderLog($data);
  581. if (!$insert) {
  582. throw new Exception('记录订单预存款支付日志出现错误');
  583. }
  584. //订单状态 置为已支付
  585. $data_order = array();
  586. $data_order['order_state'] = ORDER_STATE_PAY;
  587. $data_order['payment_time'] = time();
  588. $data_order['payment_code'] = 'predeposit';
  589. $data_order['pd_amount'] = $order_amount;
  590. $result = $model_order->editOrder($data_order,array('order_id'=>$order_info['order_id']));
  591. if (!$result) {
  592. throw new Exception('订单更新失败');
  593. }
  594. // 发送商家提醒
  595. $param = array();
  596. $param['code'] = 'new_order';
  597. $param['store_id'] = $order_info['store_id'];
  598. $param['param'] = ['order_sn' => $order_info['order_sn']];
  599. QueueClient::push('sendStoreMsg', $param);
  600. }
  601. else
  602. {
  603. //暂冻结预存款,后面还需要 API彻底完成支付
  604. if ($available_pd_amount > 0) {
  605. $data_pd['amount'] = $available_pd_amount;
  606. $model_pd->changePd('order_freeze',$data_pd);
  607. //预存款支付金额保存到订单
  608. $data_order = array();
  609. $data_order['pd_amount'] = $available_pd_amount;
  610. $result = $model_order->editOrder($data_order,array('order_id'=>$order_info['order_id']));
  611. $available_pd_amount = 0;
  612. if (!$result) {
  613. throw new Exception('订单更新失败');
  614. }
  615. }
  616. }
  617. }
  618. }
  619. /**
  620. * 预存款支付,依次循环每个订单
  621. * 如果预存款足够就单独支付了该订单,如果不足就暂时冻结,等API支付成功了再彻底扣除
  622. */
  623. public function bonusPay($order_list,$input,$buyer_info,bonus\IPriceCalculate $priceCalculate)
  624. {
  625. $member_id = $buyer_info['member_id'];
  626. $member_name = $buyer_info['member_name'];
  627. $model_pd = Model('predeposit');
  628. foreach ($order_list as $order_info)
  629. {
  630. //货到付款的订单、已经支付的订单跳过
  631. if ($order_info['payment_code'] == 'offline') continue;
  632. if ($order_info['order_state'] == ORDER_STATE_PAY) continue;
  633. $bonus_amount = unserialize($order_info['bonus_amount']);
  634. $user_bonus = $bonus_amount['user_bonus'];
  635. $room_id = $bonus_amount['room_id'];
  636. $room_bonus = $bonus_amount['room_bonus'];
  637. $priceCalculate->hold_bonus($user_bonus);
  638. room_helper::onSpendBonus($room_id,$room_bonus,$member_id);
  639. $data_pd = array();
  640. $data_pd['member_id'] = $member_id;
  641. $data_pd['member_name'] = $member_name;
  642. $data_pd['amount'] = $user_bonus;
  643. $data_pd['order_sn'] = $order_info['order_sn'];
  644. $model_pd->changeBonus('order_pay',$data_pd);
  645. }
  646. }
  647. /**
  648. * 生成支付单编号(两位随机 + 从2000-01-01 00:00:00 到现在的秒数+微秒+会员ID%1000),该值会传给第三方支付接口
  649. * 长度 =2位 + 10位 + 3位 + 3位 = 18位_
  650. * 1000个会员同一微秒提订单,重复机率为1/100
  651. * @return string
  652. */
  653. public function makePaySn($member_id) {
  654. return mt_rand(10,99)
  655. . sprintf('%010d',time() - 946656000)
  656. . sprintf('%03d', (float) microtime() * 1000)
  657. . sprintf('%03d', (int) $member_id % 1000);
  658. }
  659. /**
  660. * 订单编号生成规则,n(n>=1)个订单表对应一个支付表,
  661. * 生成订单编号(年取1位 + $pay_id取13位 + 第N个子订单取2位)
  662. * 1000个会员同一微秒提订单,重复机率为1/100
  663. * @param $pay_id 支付表自增ID
  664. * @return string
  665. */
  666. public function makeOrderSn($pay_id) {
  667. //记录生成子订单的个数,如果生成多个子订单,该值会累加
  668. static $num;
  669. if (empty($num)) {
  670. $num = 1;
  671. } else {
  672. $num ++;
  673. }
  674. return (date('y',time()) % 9+1) . sprintf('%013d', $pay_id) . sprintf('%02d', $num);
  675. }
  676. /**
  677. * 更新库存与销量
  678. *
  679. * @param array $buy_items 商品ID => 购买数量
  680. */
  681. public function editGoodsNum($buy_items) {
  682. foreach ($buy_items as $goods_id => $buy_num) {
  683. $data = array('goods_storage'=>array('exp','goods_storage-'.$buy_num),'goods_salenum'=>array('exp','goods_salenum+'.$buy_num));
  684. $result = Model('goods')->editGoods($data,array('goods_id'=>$goods_id));
  685. if (!$result) throw new Exception(L('cart_step2_submit_fail'));
  686. }
  687. }
  688. /**
  689. * 取得店铺级活动 - 每个店铺可用的满即送活动规则列表
  690. * @param unknown $store_id_array 店铺ID数组
  691. */
  692. public function getMansongRuleList($store_id_array) {
  693. if (!C('promotion_allow') || empty($store_id_array) || !is_array($store_id_array)) return array();
  694. $model_mansong = Model('p_mansong');
  695. $mansong_rule_list = array();
  696. foreach ($store_id_array as $store_id) {
  697. $store_mansong_rule = $model_mansong->getMansongInfoByStoreID($store_id);
  698. if (!empty($store_mansong_rule['rules']) && is_array($store_mansong_rule['rules'])) {
  699. foreach ($store_mansong_rule['rules'] as $rule_info) {
  700. //如果减金额 或 有赠品(在售且有库存)
  701. if (!empty($rule_info['discount']) || (!empty($rule_info['mansong_goods_name']) && !empty($rule_info['goods_storage']))) {
  702. $mansong_rule_list[$store_id][] = $this->_parseMansongRuleDesc($rule_info);
  703. }
  704. }
  705. }
  706. }
  707. return $mansong_rule_list;
  708. }
  709. /**
  710. * 取得哪些店铺有满免运费活动
  711. * @param array $store_id_array 店铺ID数组
  712. * @return array
  713. */
  714. public function getFreeFreightActiveList($store_id_array) {
  715. if (empty($store_id_array) || !is_array($store_id_array)) return array();
  716. //定义返回数组
  717. $store_free_freight_active = array();
  718. //如果商品金额未达到免运费设置下线,则需要计算运费
  719. $condition = array('store_id' => array('in',$store_id_array));
  720. $store_list = Model('store')->getStoreOnlineList($condition,null,'','store_id,store_free_price');
  721. foreach ($store_list as $store_info) {
  722. $limit_price = floatval($store_info['store_free_price']);
  723. if ($limit_price > 0) {
  724. $store_free_freight_active[$store_info['store_id']] = sprintf('满%s免运费',$limit_price);
  725. }
  726. }
  727. return $store_free_freight_active;
  728. }
  729. /**
  730. * 取得收货人地址信息
  731. * @param array $address_info
  732. * @return array
  733. */
  734. public function getReciverAddr($address_info = array()) {
  735. if (intval($address_info['dlyp_id'])) {
  736. $reciver_info['phone'] = trim($address_info['dlyp_mobile'].($address_info['dlyp_telephony'] ? ','.$address_info['dlyp_telephony'] : null),',');
  737. $reciver_info['tel_phone'] = $address_info['dlyp_telephony'];
  738. $reciver_info['mob_phone'] = $address_info['dlyp_mobile'];
  739. $reciver_info['address'] = $address_info['dlyp_area_info'].' '.$address_info['dlyp_address'];
  740. $reciver_info['area'] = $address_info['dlyp_area_info'];
  741. $reciver_info['street'] = $address_info['dlyp_address'];
  742. $reciver_info['dlyp'] = 1;
  743. $reciver_info = serialize($reciver_info);
  744. $reciver_name = $address_info['dlyp_address_name'];
  745. } else {
  746. $reciver_info['phone'] = trim($address_info['mob_phone'].($address_info['tel_phone'] ? ','.$address_info['tel_phone'] : null),',');
  747. $reciver_info['mob_phone'] = $address_info['mob_phone'];
  748. $reciver_info['tel_phone'] = $address_info['tel_phone'];
  749. $reciver_info['address'] = $address_info['area_info'].' '.$address_info['address'];
  750. $reciver_info['area'] = $address_info['area_info'];
  751. $reciver_info['street'] = $address_info['address'];
  752. $reciver_info = serialize($reciver_info);
  753. $reciver_name = $address_info['true_name'];
  754. }
  755. return array($reciver_info, $reciver_name);
  756. }
  757. /**
  758. * 整理发票信息
  759. * @param array $invoice_info 发票信息数组
  760. * @return string
  761. */
  762. public function createInvoiceData($invoice_info){
  763. //发票信息
  764. $inv = array();
  765. if ($invoice_info['inv_state'] == 1) {
  766. $inv['类型'] = '普通发票 ';
  767. $inv['抬头'] = $invoice_info['inv_title_select'] == 'person' ? '个人' : $invoice_info['inv_title'];
  768. $inv['内容'] = $invoice_info['inv_content'];
  769. } elseif (!empty($invoice_info)) {
  770. $inv['单位名称'] = $invoice_info['inv_company'];
  771. $inv['纳税人识别号'] = $invoice_info['inv_code'];
  772. $inv['注册地址'] = $invoice_info['inv_reg_addr'];
  773. $inv['注册电话'] = $invoice_info['inv_reg_phone'];
  774. $inv['开户银行'] = $invoice_info['inv_reg_bname'];
  775. $inv['银行账户'] = $invoice_info['inv_reg_baccount'];
  776. $inv['收票人姓名'] = $invoice_info['inv_rec_name'];
  777. $inv['收票人手机号'] = $invoice_info['inv_rec_mobphone'];
  778. $inv['收票人省份'] = $invoice_info['inv_rec_province'];
  779. $inv['送票地址'] = $invoice_info['inv_goto_addr'];
  780. }
  781. return !empty($inv) ? serialize($inv) : serialize(array());
  782. }
  783. /**
  784. * 计算本次下单中每个店铺订单是货到付款还是线上支付,店铺ID=>付款方式[online在线支付offline货到付款]
  785. * @param array $store_id_array 店铺ID数组
  786. * @param boolean $if_offpay 是否支持货到付款 true/false
  787. * @param string $pay_name 付款方式 online/offline
  788. * @return array
  789. */
  790. public function getStorePayTypeList($store_id_array, $if_offpay, $pay_name) {
  791. $store_pay_type_list = array();
  792. if ($pay_name == 'online') {
  793. foreach ($store_id_array as $store_id) {
  794. $store_pay_type_list[$store_id] = 'online';
  795. }
  796. } else {
  797. $offline_pay = Model('payment')->getPaymentOpenInfo(array('payment_code'=>'offline'));
  798. if ($offline_pay) {
  799. //下单里包括平台自营商品并且平台已开启货到付款
  800. $offline_store_id_array = model('store')->getOwnShopIds();
  801. foreach ($store_id_array as $store_id) {
  802. //if (in_array($store_id,$offline_store_id_array)) {
  803. $store_pay_type_list[$store_id] = 'offline';
  804. //} else {
  805. // $store_pay_type_list[$store_id] = 'online';
  806. //}
  807. }
  808. }
  809. }
  810. return $store_pay_type_list;
  811. }
  812. /**
  813. * 直接购买时返回最新的在售商品信息(需要在售)
  814. *
  815. * @param int $goods_id 所购商品ID
  816. * @param int $quantity 购买数量
  817. * @return array
  818. */
  819. private function _getGoodsOnlineInfo($goods_id,$quantity)
  820. {
  821. //取目前在售商品
  822. $goods_info = Model('goods')->getGoodsOnlineInfoAndPromotionById($goods_id);
  823. if(empty($goods_info)){
  824. return null;
  825. }
  826. if(intval($goods_info['goods_storage']) <= 0){
  827. return null;
  828. }
  829. $result = $goods_info;
  830. $result['goods_num'] = $quantity;
  831. $result['goods_id'] = $goods_id;
  832. $result['state'] = true;
  833. $result['storage_state'] = intval($goods_info['goods_storage']) < intval($quantity) ? false : true;
  834. //填充必要下标,方便后面统一使用购物车方法与模板
  835. //cart_id=goods_id,优惠套装目前只能进购物车,不能立即购买
  836. $result['cart_id'] = $goods_id;
  837. $result['bl_id'] = 0;
  838. $goods_spec = unserialize($goods_info['goods_spec']);
  839. if(empty($goods_spec)) {
  840. $goods_spec = [];
  841. }
  842. $goods_spec_array = [];
  843. foreach ($goods_spec as $value) {
  844. array_push($goods_spec_array, $value);
  845. }
  846. $result['goods_spec'] = implode(',', $goods_spec_array);
  847. return $result;
  848. }
  849. /**
  850. * 直接购买时,判断商品是不是正在抢购中,如果是,按抢购价格计算,购买数量若超过抢购规定的上限,则按抢购上限计算
  851. * @param array $goods_info
  852. */
  853. public function getGroupbuyInfo(& $goods_info = array())
  854. {
  855. if (!C('groupbuy_allow') || empty($goods_info['groupbuy_info'])) return;
  856. $groupbuy_info = $goods_info['groupbuy_info'];
  857. $goods_info['goods_price'] = $groupbuy_info['groupbuy_price'];
  858. if ($groupbuy_info['upper_limit'] && $goods_info['goods_num'] > $groupbuy_info['upper_limit']) {
  859. $goods_info['goods_num'] = $groupbuy_info['upper_limit'];
  860. }
  861. $goods_info['upper_limit'] = $groupbuy_info['upper_limit'];
  862. $goods_info['promotions_id'] = $goods_info['groupbuy_id'] = $groupbuy_info['groupbuy_id'];
  863. $goods_info['ifgroupbuy'] = true;
  864. //v3-b10
  865. //$goods_model=Model('order');
  866. //todo 抢购商品取消订单后,不让购买,如果要修改,改这行代码就可以了
  867. $ordergoods = Model()->table('order_goods')->where(array('buyer_id' => $_SESSION['member_id'], 'goods_type' => 2, 'promotions_id' => $groupbuy_info['groupbuy_id']))->sum('goods_num');
  868. if (!empty($ordergoods) && intval($ordergoods) > 0)
  869. {
  870. $tnum = intval($groupbuy_info['upper_limit']) - intval($ordergoods);//-intval($goods_info['goods_num']);
  871. if ($tnum <= 0)
  872. $goods_info = null;
  873. //return;
  874. else {
  875. if ($goods_info['goods_num'] > $tnum) {
  876. $goods_info['goods_num'] = $tnum;
  877. }
  878. }
  879. }
  880. //end
  881. }
  882. /**
  883. * 取得某商品赠品列表信息
  884. * @param array $goods_info
  885. */
  886. private function _getGoodsGiftList( & $goods_info) {
  887. if (!$goods_info['have_gift']) return ;
  888. $gift_list = Model('goods_gift')->getGoodsGiftListByGoodsId($goods_info['goods_id']);
  889. //取得赠品当前信息,如果未在售踢除,如果在售取出库存
  890. if (empty($gift_list)) return array();
  891. $model_goods = Model('goods');
  892. foreach ($gift_list as $k => $v) {
  893. $goods_online_info = $model_goods->getGoodsOnlineInfoByID($v['gift_goodsid'],'goods_storage');
  894. if (empty($goods_online_info)) {
  895. unset($gift_list[$k]);
  896. } else {
  897. $gift_list[$k]['goods_storage'] = $goods_online_info['goods_storage'];
  898. }
  899. }
  900. $goods_info['gift_list'] = $gift_list;
  901. }
  902. /**
  903. * 取商品最新的在售信息
  904. * @param unknown $cart_list
  905. * @return array
  906. */
  907. private function _getOnlineCartList($cart_list) {
  908. if (empty($cart_list) || !is_array($cart_list)) return $cart_list;
  909. //验证商品是否有效
  910. $goods_id_array = array();
  911. foreach ($cart_list as $key => $cart_info) {
  912. if (!intval($cart_info['bl_id'])) {
  913. $goods_id_array[] = $cart_info['goods_id'];
  914. }
  915. }
  916. $model_goods = Model('goods');
  917. $goods_online_list = $model_goods->getGoodsOnlineListAndPromotionByIdArray($goods_id_array);
  918. $goods_online_array = array();
  919. foreach ($goods_online_list as $goods) {
  920. $goods_online_array[$goods['goods_id']] = $goods;
  921. }
  922. foreach ((array)$cart_list as $key => $cart_info)
  923. {
  924. if (intval($cart_info['bl_id'])) continue;
  925. $cart_list[$key]['state'] = true;
  926. $cart_list[$key]['storage_state'] = true;
  927. if (in_array($cart_info['goods_id'],array_keys($goods_online_array))) {
  928. $goods_online_info = $goods_online_array[$cart_info['goods_id']];
  929. $cart_list[$key]['goods_commonid'] = $goods_online_info['goods_commonid'];
  930. $cart_list[$key]['goods_name'] = $goods_online_info['goods_mobile_name'];
  931. $cart_list[$key]['gc_id'] = $goods_online_info['gc_id'];
  932. $cart_list[$key]['goods_image'] = $goods_online_info['goods_image'];
  933. $cart_list[$key]['goods_price'] = $goods_online_info['goods_price'];
  934. $cart_list[$key]['goods_lowest_price'] = $goods_online_info['goods_lowest_price'];
  935. $cart_list[$key]['transport_id'] = $goods_online_info['transport_id'];
  936. $cart_list[$key]['goods_freight'] = $goods_online_info['goods_freight'];
  937. $cart_list[$key]['goods_vat'] = $goods_online_info['goods_vat'];
  938. $cart_list[$key]['goods_storage'] = $goods_online_info['goods_storage'];
  939. $cart_list[$key]['goods_storage_alarm'] = $goods_online_info['goods_storage_alarm'];
  940. $cart_list[$key]['is_fcode'] = $goods_online_info['is_fcode'];
  941. $cart_list[$key]['have_gift'] = $goods_online_info['have_gift'];
  942. if ($cart_info['goods_num'] > $goods_online_info['goods_storage']) {
  943. $cart_list[$key]['storage_state'] = false;
  944. }
  945. $cart_list[$key]['groupbuy_info'] = $goods_online_info['groupbuy_info'];
  946. $cart_list[$key]['xianshi_info'] = $goods_online_info['xianshi_info'];
  947. $goods_spec = unserialize($goods_online_info['goods_spec']);
  948. $goods_spec_array = array();
  949. if(empty($goods_spec)) {
  950. $goods_spec = array();
  951. }
  952. foreach ($goods_spec as $value) {
  953. array_push($goods_spec_array, $value);
  954. }
  955. $cart_list[$key]['goods_spec'] = implode(',', $goods_spec_array);
  956. } else {
  957. //如果商品下架
  958. $cart_list[$key]['state'] = false;
  959. $cart_list[$key]['storage_state'] = false;
  960. }
  961. }
  962. return $cart_list;
  963. }
  964. /**
  965. * 直接购买时,判断商品是不是正在抢购中,如果是,按抢购价格计算,购买数量若超过抢购规定的上限,则按抢购上限计算
  966. * @param array $cart_list
  967. */
  968. public function getGroupbuyCartList(& $cart_list)
  969. {
  970. if (!C('promotion_allow') || empty($cart_list)) return ;
  971. foreach ($cart_list as $key => $cart_info) {
  972. if ($cart_info['bl_id'] === '1' || empty($cart_info['groupbuy_info'])) continue;
  973. $this->getGroupbuyInfo($cart_info);
  974. $cart_list[$key] = $cart_info;
  975. }
  976. }
  977. /**
  978. * 批量判断购物车内的商品是不是限时折扣中,如果购买数量若>=规定的下限,按折扣价格计算,否则按原价计算
  979. * 并标识该商品为限时商品
  980. * @param array $cart_list
  981. */
  982. public function getXianshiCartList(&$cart_list)
  983. {
  984. if (!C('promotion_allow') || empty($cart_list)) return ;
  985. foreach ($cart_list as $key => $cart_info) {
  986. if ($cart_info['bl_id'] === '1' || empty($cart_info['xianshi_info'])) continue;
  987. $this->getXianshiInfo($cart_info, $cart_info['goods_num']);
  988. $cart_list[$key] = $cart_info;
  989. }
  990. }
  991. /**
  992. * 取得购物车商品的赠品列表[商品级赠品]
  993. *
  994. * @param array $cart_list
  995. */
  996. private function _getGiftCartList(& $cart_list) {
  997. foreach ($cart_list as $k => $cart_info) {
  998. if ($cart_info['bl_id']) continue;
  999. $this->_getGoodsGiftList($cart_info);
  1000. $cart_list[$k] = $cart_info;
  1001. }
  1002. }
  1003. /**
  1004. * 取得购买车内组合销售信息以及包含的商品及有效状态
  1005. * @param array $cart_list
  1006. */
  1007. private function _getBundlingCartList(& $cart_list)
  1008. {
  1009. if (!C('promotion_allow') || empty($cart_list)) return ;
  1010. $model_bl = Model('p_bundling');
  1011. $model_goods = Model('goods');
  1012. foreach ($cart_list as $key => $cart_info) {
  1013. if (!intval($cart_info['bl_id'])) continue;
  1014. $cart_list[$key]['state'] = true;
  1015. $cart_list[$key]['storage_state'] = true;
  1016. $bl_info = $model_bl->getBundlingInfo(array('bl_id'=>$cart_info['bl_id']));
  1017. //标志优惠套装是否处于有效状态
  1018. if (empty($bl_info) || !intval($bl_info['bl_state'])) {
  1019. $cart_list[$key]['state'] = false;
  1020. }
  1021. //取得优惠套装商品列表
  1022. $cart_list[$key]['bl_goods_list'] = $model_bl->getBundlingGoodsList(array('bl_id'=>$cart_info['bl_id']));
  1023. //取最新在售商品信息
  1024. $goods_id_array = array();
  1025. foreach ($cart_list[$key]['bl_goods_list'] as $goods_info) {
  1026. $goods_id_array[] = $goods_info['goods_id'];
  1027. }
  1028. $goods_list = $model_goods->getGoodsOnlineListAndPromotionByIdArray($goods_id_array);
  1029. $goods_online_list = array();
  1030. foreach ($goods_list as $goods_info) {
  1031. $goods_online_list[$goods_info['goods_id']] = $goods_info;
  1032. }
  1033. unset($goods_list);
  1034. //使用最新的商品名称、图片,如果一旦有商品下架,则整个套装置置为无效状态
  1035. $total_down_price = 0;
  1036. foreach ($cart_list[$key]['bl_goods_list'] as $k => $goods_info) {
  1037. if (array_key_exists($goods_info['goods_id'],$goods_online_list)) {
  1038. $goods_online_info = $goods_online_list[$goods_info['goods_id']];
  1039. //如果库存不足,标识false
  1040. if ($cart_info['goods_num'] > $goods_online_info['goods_storage']) {
  1041. $cart_list[$key]['storage_state'] = false;
  1042. }
  1043. $cart_list[$key]['bl_goods_list'][$k]['goods_id'] = $goods_online_info['goods_id'];
  1044. $cart_list[$key]['bl_goods_list'][$k]['goods_commonid'] = $goods_online_info['goods_commonid'];
  1045. $cart_list[$key]['bl_goods_list'][$k]['store_id'] = $goods_online_info['store_id'];
  1046. $cart_list[$key]['bl_goods_list'][$k]['goods_name'] = $goods_online_info['goods_name'];
  1047. $cart_list[$key]['bl_goods_list'][$k]['goods_image'] = $goods_online_info['goods_image'];
  1048. $cart_list[$key]['bl_goods_list'][$k]['goods_storage'] = $goods_online_info['goods_storage'];
  1049. $cart_list[$key]['bl_goods_list'][$k]['goods_storage_alarm'] = $goods_online_info['goods_storage_alarm'];
  1050. $cart_list[$key]['bl_goods_list'][$k]['gc_id'] = $goods_online_info['gc_id'];
  1051. //每个商品直降多少
  1052. $total_down_price += $cart_list[$key]['bl_goods_list'][$k]['down_price'] = ncPriceFormat($goods_online_info['goods_price'] - $goods_info['bl_goods_price']);
  1053. } else {
  1054. //商品已经下架
  1055. $cart_list[$key]['state'] = false;
  1056. $cart_list[$key]['storage_state'] = false;
  1057. }
  1058. }
  1059. $cart_list[$key]['down_price'] = ncPriceFormat($total_down_price);
  1060. }
  1061. }
  1062. /**
  1063. * 取得每种商品的库存
  1064. * @param array $store_cart_list 购买列表
  1065. * @param array $store_premiums_list 赠品列表
  1066. * @return array 商品ID=>库存
  1067. */
  1068. private function _getEachGoodsStorageQuantity($store_cart_list, $store_premiums_list = array())
  1069. {
  1070. if(empty($store_cart_list) || !is_array($store_cart_list)) return array();
  1071. $goods_storage_quangity = array();
  1072. foreach ($store_cart_list as $store_cart) {
  1073. foreach ($store_cart as $cart_info) {
  1074. if (!intval($cart_info['bl_id'])) {
  1075. //正常商品
  1076. $goods_storage_quangity[$cart_info['goods_id']] = $cart_info['goods_storage'];
  1077. } elseif (!empty($cart_info['bl_goods_list']) && is_array($cart_info['bl_goods_list'])) {
  1078. //优惠套装
  1079. foreach ($cart_info['bl_goods_list'] as $goods_info) {
  1080. $goods_storage_quangity[$goods_info['goods_id']] = $goods_info['goods_storage'];
  1081. }
  1082. }
  1083. }
  1084. }
  1085. //取得赠品商品的库存
  1086. if (is_array($store_premiums_list))
  1087. {
  1088. foreach ($store_premiums_list as $store_id => $goods_list) {
  1089. foreach($goods_list as $goods_info) {
  1090. if (!isset($goods_storage_quangity[$goods_info['goods_id']])) {
  1091. $goods_storage_quangity[$goods_info['goods_id']] = $goods_info['goods_storage'];
  1092. }
  1093. }
  1094. }
  1095. }
  1096. return $goods_storage_quangity;
  1097. }
  1098. /**
  1099. * 取得每种商品的购买量
  1100. * @param array $store_cart_list 购买列表
  1101. * @return array 商品ID=>购买数量
  1102. */
  1103. private function _getEachGoodsBuyQuantity($store_cart_list) {
  1104. if(empty($store_cart_list) || !is_array($store_cart_list)) return array();
  1105. $goods_buy_quangity = array();
  1106. foreach ($store_cart_list as $store_cart) {
  1107. foreach ($store_cart as $cart_info) {
  1108. if (!intval($cart_info['bl_id'])) {
  1109. //正常商品
  1110. $goods_buy_quangity[$cart_info['goods_id']] += $cart_info['goods_num'];
  1111. } elseif (!empty($cart_info['bl_goods_list']) && is_array($cart_info['bl_goods_list'])) {
  1112. //优惠套装
  1113. foreach ($cart_info['bl_goods_list'] as $goods_info) {
  1114. $goods_buy_quangity[$goods_info['goods_id']] += $cart_info['goods_num'];
  1115. }
  1116. }
  1117. }
  1118. }
  1119. return $goods_buy_quangity;
  1120. }
  1121. /**
  1122. * 得到所购买的id和数量
  1123. *
  1124. */
  1125. private function _parseItems($cart_id) {
  1126. //存放所购商品ID和数量组成的键值对
  1127. $buy_items = array();
  1128. if (is_array($cart_id)) {
  1129. foreach ($cart_id as $value) {
  1130. if (preg_match_all('/^(\d{1,10})\|(\d{1,6})$/', $value, $match)) {
  1131. $buy_items[$match[1][0]] = $match[2][0];
  1132. }
  1133. }
  1134. }
  1135. return $buy_items;
  1136. }
  1137. /**
  1138. * 拼装单条满即送规则页面描述信息
  1139. * @param array $rule_info 满即送单条规则信息
  1140. * @return string
  1141. */
  1142. private function _parseMansongRuleDesc($rule_info) {
  1143. if (empty($rule_info) || !is_array($rule_info)) return;
  1144. $discount_desc = !empty($rule_info['discount']) ? '减'.$rule_info['discount'] : '';
  1145. $goods_desc = (!empty($rule_info['mansong_goods_name']) && !empty($rule_info['goods_storage'])) ?
  1146. " 送<a href='".urlShop('goods','index',array('goods_id'=>$rule_info['goods_id']))."' title='{$rule_info['mansong_goods_name']}' target='_blank'>[赠品]</a>" : '';
  1147. return sprintf('满%s%s%s',$rule_info['price'],$discount_desc,$goods_desc);
  1148. }
  1149. }