buy_1.logic.php 56 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238
  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. public function pdPay($order_list, $input, $buyer_info)
  535. {
  536. $member_id = $buyer_info['member_id'];
  537. $member_name = $buyer_info['member_name'];
  538. $available_pd_amount = floatval($buyer_info['available_predeposit']);
  539. if ($available_pd_amount <= 0) return;
  540. $model_order = Model('order');
  541. $model_pd = Model('predeposit');
  542. foreach ($order_list as $order_info)
  543. {
  544. //货到付款的订单、已经充值卡支付的订单跳过
  545. if ($order_info['payment_code'] == 'offline') continue;
  546. if ($order_info['order_state'] == ORDER_STATE_PAY) continue;
  547. $order_amount = floatval($order_info['order_amount']) - floatval($order_info['rcb_amount']);
  548. $data_pd = array();
  549. $data_pd['member_id'] = $member_id;
  550. $data_pd['member_name'] = $member_name;
  551. $data_pd['amount'] = $order_amount;
  552. $data_pd['order_sn'] = $order_info['order_sn'];
  553. if ($available_pd_amount >= $order_amount)
  554. {
  555. //预存款立即支付,订单支付完成
  556. $model_pd->changePd('order_pay',$data_pd);
  557. $available_pd_amount -= $order_amount;
  558. //支付被冻结的充值卡
  559. $rcb_amount = floatval($order_info['rcb_amount']);
  560. if ($rcb_amount > 0) {
  561. $data_pd = array();
  562. $data_pd['member_id'] = $member_id;
  563. $data_pd['member_name'] = $member_name;
  564. $data_pd['amount'] = $rcb_amount;
  565. $data_pd['order_sn'] = $order_info['order_sn'];
  566. $model_pd->changeRcb('order_comb_pay',$data_pd);
  567. }
  568. //记录订单日志(已付款)
  569. $data = array();
  570. $data['order_id'] = $order_info['order_id'];
  571. $data['log_role'] = 'buyer';
  572. $data['log_msg'] = L('order_log_pay');
  573. $data['log_orderstate'] = ORDER_STATE_PAY;
  574. $insert = $model_order->addOrderLog($data);
  575. if (!$insert) {
  576. throw new Exception('记录订单预存款支付日志出现错误');
  577. }
  578. //订单状态 置为已支付
  579. $data_order = array();
  580. $data_order['order_state'] = ORDER_STATE_PAY;
  581. $data_order['payment_time'] = time();
  582. $data_order['payment_code'] = 'predeposit';
  583. $data_order['pd_amount'] = $order_amount;
  584. $result = $model_order->editOrder($data_order,array('order_id'=>$order_info['order_id']));
  585. if (!$result) {
  586. throw new Exception('订单更新失败');
  587. }
  588. // 发送商家提醒
  589. $param = array();
  590. $param['code'] = 'new_order';
  591. $param['store_id'] = $order_info['store_id'];
  592. $param['param'] = array(
  593. 'order_sn' => $order_info['order_sn']
  594. );
  595. QueueClient::push('sendStoreMsg', $param);
  596. }
  597. else
  598. {
  599. //暂冻结预存款,后面还需要 API彻底完成支付
  600. if ($available_pd_amount > 0) {
  601. $data_pd['amount'] = $available_pd_amount;
  602. $model_pd->changePd('order_freeze',$data_pd);
  603. //预存款支付金额保存到订单
  604. $data_order = array();
  605. $data_order['pd_amount'] = $available_pd_amount;
  606. $result = $model_order->editOrder($data_order,array('order_id'=>$order_info['order_id']));
  607. $available_pd_amount = 0;
  608. if (!$result) {
  609. throw new Exception('订单更新失败');
  610. }
  611. }
  612. }
  613. }
  614. }
  615. /**
  616. * 预存款支付,依次循环每个订单
  617. * 如果预存款足够就单独支付了该订单,如果不足就暂时冻结,等API支付成功了再彻底扣除
  618. */
  619. public function bonusPay($order_list,$input,$buyer_info,bonus\IPriceCalculate $priceCalculate)
  620. {
  621. $member_id = $buyer_info['member_id'];
  622. $member_name = $buyer_info['member_name'];
  623. $model_pd = Model('predeposit');
  624. foreach ($order_list as $order_info)
  625. {
  626. //货到付款的订单、已经支付的订单跳过
  627. if ($order_info['payment_code'] == 'offline') continue;
  628. if ($order_info['order_state'] == ORDER_STATE_PAY) continue;
  629. $bonus_amount = unserialize($order_info['bonus_amount']);
  630. $user_bonus = $bonus_amount['user_bonus'];
  631. $room_id = $bonus_amount['room_id'];
  632. $room_bonus = $bonus_amount['room_bonus'];
  633. $priceCalculate->hold_bonus($user_bonus);
  634. room_helper::onSpendBonus($room_id,$room_bonus,$member_id);
  635. $data_pd = array();
  636. $data_pd['member_id'] = $member_id;
  637. $data_pd['member_name'] = $member_name;
  638. $data_pd['amount'] = $user_bonus;
  639. $data_pd['order_sn'] = $order_info['order_sn'];
  640. $model_pd->changeBonus('order_pay',$data_pd);
  641. }
  642. }
  643. /**
  644. * 生成支付单编号(两位随机 + 从2000-01-01 00:00:00 到现在的秒数+微秒+会员ID%1000),该值会传给第三方支付接口
  645. * 长度 =2位 + 10位 + 3位 + 3位 = 18位_
  646. * 1000个会员同一微秒提订单,重复机率为1/100
  647. * @return string
  648. */
  649. public function makePaySn($member_id) {
  650. return mt_rand(10,99)
  651. . sprintf('%010d',time() - 946656000)
  652. . sprintf('%03d', (float) microtime() * 1000)
  653. . sprintf('%03d', (int) $member_id % 1000);
  654. }
  655. /**
  656. * 订单编号生成规则,n(n>=1)个订单表对应一个支付表,
  657. * 生成订单编号(年取1位 + $pay_id取13位 + 第N个子订单取2位)
  658. * 1000个会员同一微秒提订单,重复机率为1/100
  659. * @param $pay_id 支付表自增ID
  660. * @return string
  661. */
  662. public function makeOrderSn($pay_id) {
  663. //记录生成子订单的个数,如果生成多个子订单,该值会累加
  664. static $num;
  665. if (empty($num)) {
  666. $num = 1;
  667. } else {
  668. $num ++;
  669. }
  670. return (date('y',time()) % 9+1) . sprintf('%013d', $pay_id) . sprintf('%02d', $num);
  671. }
  672. /**
  673. * 更新库存与销量
  674. *
  675. * @param array $buy_items 商品ID => 购买数量
  676. */
  677. public function editGoodsNum($buy_items) {
  678. foreach ($buy_items as $goods_id => $buy_num) {
  679. $data = array('goods_storage'=>array('exp','goods_storage-'.$buy_num),'goods_salenum'=>array('exp','goods_salenum+'.$buy_num));
  680. $result = Model('goods')->editGoods($data,array('goods_id'=>$goods_id));
  681. if (!$result) throw new Exception(L('cart_step2_submit_fail'));
  682. }
  683. }
  684. /**
  685. * 取得店铺级活动 - 每个店铺可用的满即送活动规则列表
  686. * @param unknown $store_id_array 店铺ID数组
  687. */
  688. public function getMansongRuleList($store_id_array) {
  689. if (!C('promotion_allow') || empty($store_id_array) || !is_array($store_id_array)) return array();
  690. $model_mansong = Model('p_mansong');
  691. $mansong_rule_list = array();
  692. foreach ($store_id_array as $store_id) {
  693. $store_mansong_rule = $model_mansong->getMansongInfoByStoreID($store_id);
  694. if (!empty($store_mansong_rule['rules']) && is_array($store_mansong_rule['rules'])) {
  695. foreach ($store_mansong_rule['rules'] as $rule_info) {
  696. //如果减金额 或 有赠品(在售且有库存)
  697. if (!empty($rule_info['discount']) || (!empty($rule_info['mansong_goods_name']) && !empty($rule_info['goods_storage']))) {
  698. $mansong_rule_list[$store_id][] = $this->_parseMansongRuleDesc($rule_info);
  699. }
  700. }
  701. }
  702. }
  703. return $mansong_rule_list;
  704. }
  705. /**
  706. * 取得哪些店铺有满免运费活动
  707. * @param array $store_id_array 店铺ID数组
  708. * @return array
  709. */
  710. public function getFreeFreightActiveList($store_id_array) {
  711. if (empty($store_id_array) || !is_array($store_id_array)) return array();
  712. //定义返回数组
  713. $store_free_freight_active = array();
  714. //如果商品金额未达到免运费设置下线,则需要计算运费
  715. $condition = array('store_id' => array('in',$store_id_array));
  716. $store_list = Model('store')->getStoreOnlineList($condition,null,'','store_id,store_free_price');
  717. foreach ($store_list as $store_info) {
  718. $limit_price = floatval($store_info['store_free_price']);
  719. if ($limit_price > 0) {
  720. $store_free_freight_active[$store_info['store_id']] = sprintf('满%s免运费',$limit_price);
  721. }
  722. }
  723. return $store_free_freight_active;
  724. }
  725. /**
  726. * 取得收货人地址信息
  727. * @param array $address_info
  728. * @return array
  729. */
  730. public function getReciverAddr($address_info = array()) {
  731. if (intval($address_info['dlyp_id'])) {
  732. $reciver_info['phone'] = trim($address_info['dlyp_mobile'].($address_info['dlyp_telephony'] ? ','.$address_info['dlyp_telephony'] : null),',');
  733. $reciver_info['tel_phone'] = $address_info['dlyp_telephony'];
  734. $reciver_info['mob_phone'] = $address_info['dlyp_mobile'];
  735. $reciver_info['address'] = $address_info['dlyp_area_info'].' '.$address_info['dlyp_address'];
  736. $reciver_info['area'] = $address_info['dlyp_area_info'];
  737. $reciver_info['street'] = $address_info['dlyp_address'];
  738. $reciver_info['dlyp'] = 1;
  739. $reciver_info = serialize($reciver_info);
  740. $reciver_name = $address_info['dlyp_address_name'];
  741. } else {
  742. $reciver_info['phone'] = trim($address_info['mob_phone'].($address_info['tel_phone'] ? ','.$address_info['tel_phone'] : null),',');
  743. $reciver_info['mob_phone'] = $address_info['mob_phone'];
  744. $reciver_info['tel_phone'] = $address_info['tel_phone'];
  745. $reciver_info['address'] = $address_info['area_info'].' '.$address_info['address'];
  746. $reciver_info['area'] = $address_info['area_info'];
  747. $reciver_info['street'] = $address_info['address'];
  748. $reciver_info = serialize($reciver_info);
  749. $reciver_name = $address_info['true_name'];
  750. }
  751. return array($reciver_info, $reciver_name);
  752. }
  753. /**
  754. * 整理发票信息
  755. * @param array $invoice_info 发票信息数组
  756. * @return string
  757. */
  758. public function createInvoiceData($invoice_info){
  759. //发票信息
  760. $inv = array();
  761. if ($invoice_info['inv_state'] == 1) {
  762. $inv['类型'] = '普通发票 ';
  763. $inv['抬头'] = $invoice_info['inv_title_select'] == 'person' ? '个人' : $invoice_info['inv_title'];
  764. $inv['内容'] = $invoice_info['inv_content'];
  765. } elseif (!empty($invoice_info)) {
  766. $inv['单位名称'] = $invoice_info['inv_company'];
  767. $inv['纳税人识别号'] = $invoice_info['inv_code'];
  768. $inv['注册地址'] = $invoice_info['inv_reg_addr'];
  769. $inv['注册电话'] = $invoice_info['inv_reg_phone'];
  770. $inv['开户银行'] = $invoice_info['inv_reg_bname'];
  771. $inv['银行账户'] = $invoice_info['inv_reg_baccount'];
  772. $inv['收票人姓名'] = $invoice_info['inv_rec_name'];
  773. $inv['收票人手机号'] = $invoice_info['inv_rec_mobphone'];
  774. $inv['收票人省份'] = $invoice_info['inv_rec_province'];
  775. $inv['送票地址'] = $invoice_info['inv_goto_addr'];
  776. }
  777. return !empty($inv) ? serialize($inv) : serialize(array());
  778. }
  779. /**
  780. * 计算本次下单中每个店铺订单是货到付款还是线上支付,店铺ID=>付款方式[online在线支付offline货到付款]
  781. * @param array $store_id_array 店铺ID数组
  782. * @param boolean $if_offpay 是否支持货到付款 true/false
  783. * @param string $pay_name 付款方式 online/offline
  784. * @return array
  785. */
  786. public function getStorePayTypeList($store_id_array, $if_offpay, $pay_name) {
  787. $store_pay_type_list = array();
  788. if ($pay_name == 'online') {
  789. foreach ($store_id_array as $store_id) {
  790. $store_pay_type_list[$store_id] = 'online';
  791. }
  792. } else {
  793. $offline_pay = Model('payment')->getPaymentOpenInfo(array('payment_code'=>'offline'));
  794. if ($offline_pay) {
  795. //下单里包括平台自营商品并且平台已开启货到付款
  796. $offline_store_id_array = model('store')->getOwnShopIds();
  797. foreach ($store_id_array as $store_id) {
  798. //if (in_array($store_id,$offline_store_id_array)) {
  799. $store_pay_type_list[$store_id] = 'offline';
  800. //} else {
  801. // $store_pay_type_list[$store_id] = 'online';
  802. //}
  803. }
  804. }
  805. }
  806. return $store_pay_type_list;
  807. }
  808. /**
  809. * 直接购买时返回最新的在售商品信息(需要在售)
  810. *
  811. * @param int $goods_id 所购商品ID
  812. * @param int $quantity 购买数量
  813. * @return array
  814. */
  815. private function _getGoodsOnlineInfo($goods_id,$quantity)
  816. {
  817. //取目前在售商品
  818. $goods_info = Model('goods')->getGoodsOnlineInfoAndPromotionById($goods_id);
  819. if(empty($goods_info)){
  820. return null;
  821. }
  822. if(intval($goods_info['goods_storage']) <= 0){
  823. return null;
  824. }
  825. $result = $goods_info;
  826. $result['goods_num'] = $quantity;
  827. $result['goods_id'] = $goods_id;
  828. $result['state'] = true;
  829. $result['storage_state'] = intval($goods_info['goods_storage']) < intval($quantity) ? false : true;
  830. //填充必要下标,方便后面统一使用购物车方法与模板
  831. //cart_id=goods_id,优惠套装目前只能进购物车,不能立即购买
  832. $result['cart_id'] = $goods_id;
  833. $result['bl_id'] = 0;
  834. return $result;
  835. }
  836. /**
  837. * 直接购买时,判断商品是不是正在抢购中,如果是,按抢购价格计算,购买数量若超过抢购规定的上限,则按抢购上限计算
  838. * @param array $goods_info
  839. */
  840. public function getGroupbuyInfo(& $goods_info = array())
  841. {
  842. if (!C('groupbuy_allow') || empty($goods_info['groupbuy_info'])) return;
  843. $groupbuy_info = $goods_info['groupbuy_info'];
  844. $goods_info['goods_price'] = $groupbuy_info['groupbuy_price'];
  845. if ($groupbuy_info['upper_limit'] && $goods_info['goods_num'] > $groupbuy_info['upper_limit']) {
  846. $goods_info['goods_num'] = $groupbuy_info['upper_limit'];
  847. }
  848. $goods_info['upper_limit'] = $groupbuy_info['upper_limit'];
  849. $goods_info['promotions_id'] = $goods_info['groupbuy_id'] = $groupbuy_info['groupbuy_id'];
  850. $goods_info['ifgroupbuy'] = true;
  851. //v3-b10
  852. //$goods_model=Model('order');
  853. //todo 抢购商品取消订单后,不让购买,如果要修改,改这行代码就可以了
  854. $ordergoods = Model()->table('order_goods')->where(array('buyer_id' => $_SESSION['member_id'], 'goods_type' => 2, 'promotions_id' => $groupbuy_info['groupbuy_id']))->sum('goods_num');
  855. if (!empty($ordergoods) && intval($ordergoods) > 0)
  856. {
  857. $tnum = intval($groupbuy_info['upper_limit']) - intval($ordergoods);//-intval($goods_info['goods_num']);
  858. if ($tnum <= 0)
  859. $goods_info = null;
  860. //return;
  861. else {
  862. if ($goods_info['goods_num'] > $tnum) {
  863. $goods_info['goods_num'] = $tnum;
  864. }
  865. }
  866. }
  867. //end
  868. }
  869. /**
  870. * 取得某商品赠品列表信息
  871. * @param array $goods_info
  872. */
  873. private function _getGoodsGiftList( & $goods_info) {
  874. if (!$goods_info['have_gift']) return ;
  875. $gift_list = Model('goods_gift')->getGoodsGiftListByGoodsId($goods_info['goods_id']);
  876. //取得赠品当前信息,如果未在售踢除,如果在售取出库存
  877. if (empty($gift_list)) return array();
  878. $model_goods = Model('goods');
  879. foreach ($gift_list as $k => $v) {
  880. $goods_online_info = $model_goods->getGoodsOnlineInfoByID($v['gift_goodsid'],'goods_storage');
  881. if (empty($goods_online_info)) {
  882. unset($gift_list[$k]);
  883. } else {
  884. $gift_list[$k]['goods_storage'] = $goods_online_info['goods_storage'];
  885. }
  886. }
  887. $goods_info['gift_list'] = $gift_list;
  888. }
  889. /**
  890. * 取商品最新的在售信息
  891. * @param unknown $cart_list
  892. * @return array
  893. */
  894. private function _getOnlineCartList($cart_list) {
  895. if (empty($cart_list) || !is_array($cart_list)) return $cart_list;
  896. //验证商品是否有效
  897. $goods_id_array = array();
  898. foreach ($cart_list as $key => $cart_info) {
  899. if (!intval($cart_info['bl_id'])) {
  900. $goods_id_array[] = $cart_info['goods_id'];
  901. }
  902. }
  903. $model_goods = Model('goods');
  904. $goods_online_list = $model_goods->getGoodsOnlineListAndPromotionByIdArray($goods_id_array);
  905. $goods_online_array = array();
  906. foreach ($goods_online_list as $goods) {
  907. $goods_online_array[$goods['goods_id']] = $goods;
  908. }
  909. foreach ((array)$cart_list as $key => $cart_info)
  910. {
  911. if (intval($cart_info['bl_id'])) continue;
  912. $cart_list[$key]['state'] = true;
  913. $cart_list[$key]['storage_state'] = true;
  914. if (in_array($cart_info['goods_id'],array_keys($goods_online_array))) {
  915. $goods_online_info = $goods_online_array[$cart_info['goods_id']];
  916. $cart_list[$key]['goods_commonid'] = $goods_online_info['goods_commonid'];
  917. $cart_list[$key]['goods_name'] = $goods_online_info['goods_mobile_name'];
  918. $cart_list[$key]['gc_id'] = $goods_online_info['gc_id'];
  919. $cart_list[$key]['goods_image'] = $goods_online_info['goods_image'];
  920. $cart_list[$key]['goods_price'] = $goods_online_info['goods_price'];
  921. $cart_list[$key]['goods_lowest_price'] = $goods_online_info['goods_lowest_price'];
  922. $cart_list[$key]['transport_id'] = $goods_online_info['transport_id'];
  923. $cart_list[$key]['goods_freight'] = $goods_online_info['goods_freight'];
  924. $cart_list[$key]['goods_vat'] = $goods_online_info['goods_vat'];
  925. $cart_list[$key]['goods_storage'] = $goods_online_info['goods_storage'];
  926. $cart_list[$key]['goods_storage_alarm'] = $goods_online_info['goods_storage_alarm'];
  927. $cart_list[$key]['is_fcode'] = $goods_online_info['is_fcode'];
  928. $cart_list[$key]['have_gift'] = $goods_online_info['have_gift'];
  929. if ($cart_info['goods_num'] > $goods_online_info['goods_storage']) {
  930. $cart_list[$key]['storage_state'] = false;
  931. }
  932. $cart_list[$key]['groupbuy_info'] = $goods_online_info['groupbuy_info'];
  933. $cart_list[$key]['xianshi_info'] = $goods_online_info['xianshi_info'];
  934. $goods_spec = unserialize($goods_online_info['goods_spec']);
  935. $goods_spec_array = array();
  936. if(empty($goods_spec)) {
  937. $goods_spec = array();
  938. }
  939. foreach ($goods_spec as $value) {
  940. array_push($goods_spec_array, $value);
  941. }
  942. $cart_list[$key]['goods_spec'] = implode(',', $goods_spec_array);
  943. } else {
  944. //如果商品下架
  945. $cart_list[$key]['state'] = false;
  946. $cart_list[$key]['storage_state'] = false;
  947. }
  948. }
  949. return $cart_list;
  950. }
  951. /**
  952. * 直接购买时,判断商品是不是正在抢购中,如果是,按抢购价格计算,购买数量若超过抢购规定的上限,则按抢购上限计算
  953. * @param array $cart_list
  954. */
  955. public function getGroupbuyCartList(& $cart_list)
  956. {
  957. if (!C('promotion_allow') || empty($cart_list)) return ;
  958. foreach ($cart_list as $key => $cart_info) {
  959. if ($cart_info['bl_id'] === '1' || empty($cart_info['groupbuy_info'])) continue;
  960. $this->getGroupbuyInfo($cart_info);
  961. $cart_list[$key] = $cart_info;
  962. }
  963. }
  964. /**
  965. * 批量判断购物车内的商品是不是限时折扣中,如果购买数量若>=规定的下限,按折扣价格计算,否则按原价计算
  966. * 并标识该商品为限时商品
  967. * @param array $cart_list
  968. */
  969. public function getXianshiCartList(&$cart_list)
  970. {
  971. if (!C('promotion_allow') || empty($cart_list)) return ;
  972. foreach ($cart_list as $key => $cart_info) {
  973. if ($cart_info['bl_id'] === '1' || empty($cart_info['xianshi_info'])) continue;
  974. $this->getXianshiInfo($cart_info, $cart_info['goods_num']);
  975. $cart_list[$key] = $cart_info;
  976. }
  977. }
  978. /**
  979. * 取得购物车商品的赠品列表[商品级赠品]
  980. *
  981. * @param array $cart_list
  982. */
  983. private function _getGiftCartList(& $cart_list) {
  984. foreach ($cart_list as $k => $cart_info) {
  985. if ($cart_info['bl_id']) continue;
  986. $this->_getGoodsGiftList($cart_info);
  987. $cart_list[$k] = $cart_info;
  988. }
  989. }
  990. /**
  991. * 取得购买车内组合销售信息以及包含的商品及有效状态
  992. * @param array $cart_list
  993. */
  994. private function _getBundlingCartList(& $cart_list)
  995. {
  996. if (!C('promotion_allow') || empty($cart_list)) return ;
  997. $model_bl = Model('p_bundling');
  998. $model_goods = Model('goods');
  999. foreach ($cart_list as $key => $cart_info) {
  1000. if (!intval($cart_info['bl_id'])) continue;
  1001. $cart_list[$key]['state'] = true;
  1002. $cart_list[$key]['storage_state'] = true;
  1003. $bl_info = $model_bl->getBundlingInfo(array('bl_id'=>$cart_info['bl_id']));
  1004. //标志优惠套装是否处于有效状态
  1005. if (empty($bl_info) || !intval($bl_info['bl_state'])) {
  1006. $cart_list[$key]['state'] = false;
  1007. }
  1008. //取得优惠套装商品列表
  1009. $cart_list[$key]['bl_goods_list'] = $model_bl->getBundlingGoodsList(array('bl_id'=>$cart_info['bl_id']));
  1010. //取最新在售商品信息
  1011. $goods_id_array = array();
  1012. foreach ($cart_list[$key]['bl_goods_list'] as $goods_info) {
  1013. $goods_id_array[] = $goods_info['goods_id'];
  1014. }
  1015. $goods_list = $model_goods->getGoodsOnlineListAndPromotionByIdArray($goods_id_array);
  1016. $goods_online_list = array();
  1017. foreach ($goods_list as $goods_info) {
  1018. $goods_online_list[$goods_info['goods_id']] = $goods_info;
  1019. }
  1020. unset($goods_list);
  1021. //使用最新的商品名称、图片,如果一旦有商品下架,则整个套装置置为无效状态
  1022. $total_down_price = 0;
  1023. foreach ($cart_list[$key]['bl_goods_list'] as $k => $goods_info) {
  1024. if (array_key_exists($goods_info['goods_id'],$goods_online_list)) {
  1025. $goods_online_info = $goods_online_list[$goods_info['goods_id']];
  1026. //如果库存不足,标识false
  1027. if ($cart_info['goods_num'] > $goods_online_info['goods_storage']) {
  1028. $cart_list[$key]['storage_state'] = false;
  1029. }
  1030. $cart_list[$key]['bl_goods_list'][$k]['goods_id'] = $goods_online_info['goods_id'];
  1031. $cart_list[$key]['bl_goods_list'][$k]['goods_commonid'] = $goods_online_info['goods_commonid'];
  1032. $cart_list[$key]['bl_goods_list'][$k]['store_id'] = $goods_online_info['store_id'];
  1033. $cart_list[$key]['bl_goods_list'][$k]['goods_name'] = $goods_online_info['goods_name'];
  1034. $cart_list[$key]['bl_goods_list'][$k]['goods_image'] = $goods_online_info['goods_image'];
  1035. $cart_list[$key]['bl_goods_list'][$k]['goods_storage'] = $goods_online_info['goods_storage'];
  1036. $cart_list[$key]['bl_goods_list'][$k]['goods_storage_alarm'] = $goods_online_info['goods_storage_alarm'];
  1037. $cart_list[$key]['bl_goods_list'][$k]['gc_id'] = $goods_online_info['gc_id'];
  1038. //每个商品直降多少
  1039. $total_down_price += $cart_list[$key]['bl_goods_list'][$k]['down_price'] = ncPriceFormat($goods_online_info['goods_price'] - $goods_info['bl_goods_price']);
  1040. } else {
  1041. //商品已经下架
  1042. $cart_list[$key]['state'] = false;
  1043. $cart_list[$key]['storage_state'] = false;
  1044. }
  1045. }
  1046. $cart_list[$key]['down_price'] = ncPriceFormat($total_down_price);
  1047. }
  1048. }
  1049. /**
  1050. * 取得每种商品的库存
  1051. * @param array $store_cart_list 购买列表
  1052. * @param array $store_premiums_list 赠品列表
  1053. * @return array 商品ID=>库存
  1054. */
  1055. private function _getEachGoodsStorageQuantity($store_cart_list, $store_premiums_list = array())
  1056. {
  1057. if(empty($store_cart_list) || !is_array($store_cart_list)) return array();
  1058. $goods_storage_quangity = array();
  1059. foreach ($store_cart_list as $store_cart) {
  1060. foreach ($store_cart as $cart_info) {
  1061. if (!intval($cart_info['bl_id'])) {
  1062. //正常商品
  1063. $goods_storage_quangity[$cart_info['goods_id']] = $cart_info['goods_storage'];
  1064. } elseif (!empty($cart_info['bl_goods_list']) && is_array($cart_info['bl_goods_list'])) {
  1065. //优惠套装
  1066. foreach ($cart_info['bl_goods_list'] as $goods_info) {
  1067. $goods_storage_quangity[$goods_info['goods_id']] = $goods_info['goods_storage'];
  1068. }
  1069. }
  1070. }
  1071. }
  1072. //取得赠品商品的库存
  1073. if (is_array($store_premiums_list))
  1074. {
  1075. foreach ($store_premiums_list as $store_id => $goods_list) {
  1076. foreach($goods_list as $goods_info) {
  1077. if (!isset($goods_storage_quangity[$goods_info['goods_id']])) {
  1078. $goods_storage_quangity[$goods_info['goods_id']] = $goods_info['goods_storage'];
  1079. }
  1080. }
  1081. }
  1082. }
  1083. return $goods_storage_quangity;
  1084. }
  1085. /**
  1086. * 取得每种商品的购买量
  1087. * @param array $store_cart_list 购买列表
  1088. * @return array 商品ID=>购买数量
  1089. */
  1090. private function _getEachGoodsBuyQuantity($store_cart_list) {
  1091. if(empty($store_cart_list) || !is_array($store_cart_list)) return array();
  1092. $goods_buy_quangity = array();
  1093. foreach ($store_cart_list as $store_cart) {
  1094. foreach ($store_cart as $cart_info) {
  1095. if (!intval($cart_info['bl_id'])) {
  1096. //正常商品
  1097. $goods_buy_quangity[$cart_info['goods_id']] += $cart_info['goods_num'];
  1098. } elseif (!empty($cart_info['bl_goods_list']) && is_array($cart_info['bl_goods_list'])) {
  1099. //优惠套装
  1100. foreach ($cart_info['bl_goods_list'] as $goods_info) {
  1101. $goods_buy_quangity[$goods_info['goods_id']] += $cart_info['goods_num'];
  1102. }
  1103. }
  1104. }
  1105. }
  1106. return $goods_buy_quangity;
  1107. }
  1108. /**
  1109. * 得到所购买的id和数量
  1110. *
  1111. */
  1112. private function _parseItems($cart_id) {
  1113. //存放所购商品ID和数量组成的键值对
  1114. $buy_items = array();
  1115. if (is_array($cart_id)) {
  1116. foreach ($cart_id as $value) {
  1117. if (preg_match_all('/^(\d{1,10})\|(\d{1,6})$/', $value, $match)) {
  1118. $buy_items[$match[1][0]] = $match[2][0];
  1119. }
  1120. }
  1121. }
  1122. return $buy_items;
  1123. }
  1124. /**
  1125. * 拼装单条满即送规则页面描述信息
  1126. * @param array $rule_info 满即送单条规则信息
  1127. * @return string
  1128. */
  1129. private function _parseMansongRuleDesc($rule_info) {
  1130. if (empty($rule_info) || !is_array($rule_info)) return;
  1131. $discount_desc = !empty($rule_info['discount']) ? '减'.$rule_info['discount'] : '';
  1132. $goods_desc = (!empty($rule_info['mansong_goods_name']) && !empty($rule_info['goods_storage'])) ?
  1133. " 送<a href='".urlShop('goods','index',array('goods_id'=>$rule_info['goods_id']))."' title='{$rule_info['mansong_goods_name']}' target='_blank'>[赠品]</a>" : '';
  1134. return sprintf('满%s%s%s',$rule_info['price'],$discount_desc,$goods_desc);
  1135. }
  1136. }