MCart.Class.php 151 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519
  1. <?php
  2. /**
  3. * 购物车
  4. * Created by PhpStorm.
  5. * User: XiaoMing
  6. * Date: 2019/11/6
  7. * Time: 16:20
  8. */
  9. namespace JinDouYun\Model\Cart;
  10. use JinDouYun\Dao\Cashier\DCashierCustomerPrice;
  11. use JinDouYun\Dao\System\DDeliveryRule;
  12. use JinDouYun\Model\Cashier\MCashierSettings;
  13. use JinDouYun\Model\Market\MComBinPackage;
  14. use Mall\Framework\Core\ErrorCode;
  15. use Mall\Framework\Core\ResultWrapper;
  16. use Mall\Framework\Core\StatusCode;
  17. use JinDouYun\Dao\Cart\DCart;
  18. use JinDouYun\Cache\ActivityLimitCache;
  19. use JinDouYun\Controller\Common\Logger;
  20. use JinDouYun\Model\Market\MVipCard;
  21. use JinDouYun\Model\System\MBasicSetup;
  22. use JinDouYun\Model\Market\MActivity;
  23. use JinDouYun\Model\Market\MUserCoupon;
  24. use JinDouYun\Model\System\MEnterpriseBindPayment;
  25. use JinDouYun\Model\Customer\MCustomer;
  26. use JinDouYun\Model\Customer\MShippingAddress;
  27. use JinDouYun\Model\GoodsManage\MSku;
  28. use JinDouYun\Model\Price\MPrice;
  29. use JinDouYun\Model\Stock\MInventory;
  30. use JinDouYun\Model\SysAreaChina\MSysAreaChina;
  31. use JinDouYun\Model\System\MDeliverySetting;
  32. use JinDouYun\Model\Goods\MGoods;
  33. use JinDouYun\Model\Shop\MShop;
  34. use JinDouYun\Cache\CustomerCache;
  35. use JinDouYun\Cache\GoodsBasicRelevant;
  36. use JinDouYun\Dao\Customer\DCustomer;
  37. use JinDouYun\Model\Cart\MPreferentialProcess;
  38. class MCart
  39. {
  40. /**
  41. * @var DCart
  42. */
  43. private $objDCart;
  44. /**
  45. * @var int 用户id
  46. */
  47. private $onlineUserId;
  48. /**
  49. * @var int 企业id
  50. */
  51. private $onlineEnterpriseId;
  52. /**
  53. * @var MCustomer
  54. */
  55. private $objMCustomer;
  56. /**
  57. * @var ActivityLimitCache
  58. */
  59. private $objActivityLimitCache;
  60. /**
  61. * @var bool
  62. */
  63. private $isFront;//是否是前台调用此Model 前台=>true
  64. /**
  65. * @var int 客户id
  66. */
  67. private $customerId;//用户对应的客户id
  68. /**
  69. * @var MActivity
  70. */
  71. private $objMActivity;
  72. /**
  73. * @var array 失效商品集合
  74. */
  75. private $invalidData = [];
  76. /**
  77. * @var array 操作类型
  78. */
  79. private static $type = [
  80. 'single' => 1,//单个商品
  81. 'shop' => 2,//按店铺
  82. 'all' => 3,//全选
  83. ];
  84. /**
  85. * @var array 阶梯价
  86. */
  87. private static $enabledLadder = [
  88. 'open' => 1,//启用阶梯价
  89. 'close' => 0,//未启用阶梯价
  90. ];
  91. /**
  92. * @var int[] 立即购买来源业务
  93. */
  94. public static $sourceType = [
  95. 'comBin' => 1,//组合套餐
  96. 'goods' => 2, // 普通商品
  97. ];
  98. /**
  99. * @var MSku
  100. */
  101. private $objMSku;
  102. /**
  103. * @var MGoods
  104. */
  105. private $objMGoods;
  106. /**
  107. * @var MBasicSetup
  108. */
  109. private $objMBasicSetup;
  110. /**
  111. * @var int
  112. */
  113. private $onlineUserDefaultDeliveryType = null;
  114. /**
  115. * @var float
  116. */
  117. private $onlineUserAddressCode = null;
  118. /**
  119. * @var float
  120. */
  121. private $freeExpressPrice = null;
  122. /**
  123. * @var string
  124. */
  125. private $cashierUId;
  126. /**
  127. * @var DCustomer
  128. */
  129. private $objDCustomer;
  130. /**
  131. * @var MCashierSettings
  132. */
  133. private $objMCashierSettings;
  134. /**
  135. * @var DCashierCustomerPrice
  136. */
  137. private $objDCashierCustomerPrice;
  138. /**
  139. * @var boolean
  140. */
  141. private $isCashier;
  142. /**
  143. * @var float 运费金额
  144. */
  145. private $expressMoney = 0;
  146. /**
  147. * @var array 店铺运费
  148. */
  149. private $expressShopMoney = [];
  150. /**
  151. * @var array
  152. */
  153. private $selectStateMap = [];
  154. /**
  155. * @var bool
  156. */
  157. private $preSale;
  158. /**
  159. * MCart constructor.
  160. * @param $onlineUserId
  161. * @param $onlineEnterpriseId
  162. * @param bool $isFront
  163. * @param string $areaCode
  164. * @param string $cashierUId
  165. * @param boolean $isCashier
  166. * @throws \Exception
  167. */
  168. public function __construct($onlineUserId, $onlineEnterpriseId, $isFront = false, $areaCode = '', $cashierUId = '', $isCashier = false)
  169. {
  170. $this->isCashier = $isCashier;
  171. $this->cashierUId = $cashierUId;
  172. $this->isFront = $isFront;
  173. $this->onlineUserId = $onlineUserId;
  174. $this->onlineEnterpriseId = $onlineEnterpriseId;
  175. $this->objDCart = new DCart('default');
  176. $this->objDCart->setTable($this->objDCart->get_Table() . '_' . $onlineEnterpriseId);
  177. $this->objMCustomer = new MCustomer($this->onlineEnterpriseId, $this->onlineUserId);
  178. $this->objActivityLimitCache = new ActivityLimitCache($this->onlineEnterpriseId);
  179. $this->objMActivity = new MActivity($this->onlineUserId, $this->onlineEnterpriseId, $areaCode);
  180. $this->objMSku = new MSku($this->onlineUserId, $this->onlineEnterpriseId);
  181. $this->objMGoods = new MGoods($this->onlineEnterpriseId, $this->isFront, $this->onlineUserId);
  182. $this->objMBasicSetup = new MBasicSetup($this->onlineEnterpriseId);
  183. $this->objDCustomer = new DCustomer();
  184. $this->objDCustomer->setTable('qianniao_customer_' . $this->onlineEnterpriseId);
  185. $this->objMCashierSettings = new MCashierSettings($this->onlineUserId, $this->onlineEnterpriseId);
  186. $this->objDCashierCustomerPrice = new DCashierCustomerPrice();
  187. $this->objDCashierCustomerPrice->setTable('qianniao_cashier_customer_price_' . $this->onlineEnterpriseId);
  188. self::getCustomerInfo();
  189. }
  190. /**
  191. * Doc: (des="")
  192. * User: XMing
  193. * Date: 2020/12/25
  194. * Time: 11:50 上午
  195. * @return ResultWrapper
  196. */
  197. public function getPreSale(): ResultWrapper
  198. {
  199. $objMBasicSetup = new MBasicSetup($this->onlineEnterpriseId);
  200. $setResult = $objMBasicSetup->getBasicField('preSale');
  201. $this->preSale = StatusCode::$delete;
  202. if (!$setResult->isSuccess()) {
  203. return ResultWrapper::fail($setResult->getData(), $setResult->getErrorCode());
  204. }
  205. $set = (array)$setResult->getData();
  206. $this->preSale = getArrayItem($set, 'preSale', StatusCode::$delete);
  207. if (empty($this->preSale)) $this->preSale = StatusCode::$delete;
  208. return ResultWrapper::success($set);
  209. }
  210. /**
  211. * 获取客户id
  212. */
  213. private function getCustomerInfo()
  214. {
  215. if ($this->isFront === false) {
  216. $this->customerId = $this->onlineUserId;//TODO()
  217. } else {
  218. $this->customerId = $this->objMCustomer->getCustomerIdByUserCenterId($this->onlineUserId);
  219. }
  220. }
  221. /**
  222. * 根据限购数量对数据进行分组
  223. * @param $data
  224. * @return array
  225. */
  226. private function checkLimitGroup($data)
  227. {
  228. if (empty($data)) return $data;
  229. $allMapping = [];
  230. foreach ($data as &$goods) {
  231. if ($goods['isActivityPrice'] == StatusCode::$standard) {
  232. $userLimit = $this->objActivityLimitCache->getLimit($goods['activityId'], $goods['goodsId'], $goods['skuId'], $this->onlineUserId);//用户限购数量
  233. $allowNum = $goods['limitNum'] - $userLimit;//还可以购买的数量
  234. $buyNum = $goods['buyNum'];//购买数量
  235. if ($buyNum >= $allowNum) {
  236. //拆分
  237. $goods['buyNum'] = $allowNum;
  238. $allMapping[] = $goods;
  239. $newGoods = $goods;
  240. if ($buyNum - $allowNum > 0) {
  241. $newGoods['buyNum'] = $buyNum - $allowNum;
  242. $newGoods['activityId'] = 0;//将活动id重置为0
  243. $newGoods['isActivityPrice'] = StatusCode::$delete;
  244. $allMapping[] = $newGoods;
  245. }
  246. } else {
  247. //限购等于加入数量
  248. $allMapping[] = $goods;
  249. }
  250. } else {
  251. $allMapping[] = $goods;
  252. }
  253. }
  254. return $allMapping;
  255. }
  256. /**
  257. * Doc: (des="小程序端加入购物车,最新")
  258. * User: XMing
  259. * Date: 2020/8/4
  260. * Time: 2:35 下午
  261. * @param array $params
  262. * @return ResultWrapper
  263. * @throws \Exception
  264. */
  265. public function addCartApi(array $params)
  266. {
  267. if (empty( $params['goodsData'] )) {
  268. return ResultWrapper::fail('加入购物车数据为空', ErrorCode::$paramError);
  269. }
  270. foreach ( $params['goodsData'] as $key => $data ){
  271. // 从商品表查询商品信息
  272. $goodsInfoResult = $this->objMGoods->getGoodsInfo($data['goodsId']);
  273. if (!$goodsInfoResult->isSuccess()) {
  274. return ResultWrapper::fail($goodsInfoResult->getData(), $goodsInfoResult->getErrorCode());
  275. }
  276. $goodsInfo = $goodsInfoResult->getData();
  277. // 验证此商品是否有效
  278. $checkResult = self::checkGoods($goodsInfo, $data, 'add');
  279. unset($data);
  280. if (!$checkResult->isSuccess()) {
  281. return ResultWrapper::fail($checkResult->getData(), $checkResult->getErrorCode());
  282. }
  283. $data = $checkResult->getData();
  284. // 判断此商品是否是活动商品
  285. $addMap = [
  286. 'goodsBasicId' => $data['goodsBasicId'],
  287. 'goodsId' => $data['goodsId'],
  288. 'skuId' => $data['skuId'],
  289. 'buyNum' => $data['buyNum'],
  290. 'shopId' => $data['shopId'],
  291. 'source' => $data['source'],
  292. 'setNum' => $data['setNum'],
  293. 'activityId' => $data['activityId'],
  294. 'goodsCode' => $data['goodsCode'],
  295. 'warehouseId' => $data['warehouseId']
  296. ];
  297. // 将加入购物车的数据进行分组[old=>在购物车中已存在,now=>新加入购物车商品]
  298. $groupResult = self::existCartAndGroup([$addMap]);
  299. if (!$groupResult->isSuccess()) {
  300. return ResultWrapper::fail($groupResult->getData(), $groupResult->getErrorCode());
  301. }
  302. $groupData = $groupResult->getData();
  303. if (empty($groupData)) {
  304. return ResultWrapper::fail('数组分组失败', ErrorCode::$paramError);
  305. }
  306. // 在购物车中已存在的商品
  307. $updates = isset($groupData['old']) ? $groupData['old'] : [];
  308. // 新加入购物车商品
  309. $inserts = isset($groupData['now']) ? $groupData['now'] : [];
  310. // 操作数据库
  311. $this->objDCart->beginTransaction();
  312. // 需要更新的数据
  313. if (!empty($updates)) {
  314. $update = array_shift($updates);
  315. $tableName = 'qianniao_cart_' . $this->onlineEnterpriseId;
  316. $sql = 'UPDATE ' . $tableName . ' SET buyNum = buyNum+' . $update['buyNum'] . '
  317. WHERE goodsId = ' . $update['goodsId'] . ' AND
  318. skuId = ' . $update['skuId'] . ' AND
  319. userCenterId = ' . $this->onlineUserId . ' AND
  320. activityId = ' . $update['activityId'];
  321. $updateRes = $this->objDCart->query($sql);
  322. if ($updateRes === false) {
  323. $this->objDCart->rollBack();
  324. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  325. }
  326. }
  327. // 需要插入的数据
  328. if (!empty($inserts)) {
  329. $insert = array_shift($inserts);
  330. $insertMap = [
  331. 'userCenterId' => $this->onlineUserId,
  332. 'selection' => StatusCode::$standard,
  333. 'goodsBasicId' => $insert['goodsBasicId'],
  334. 'goodsId' => $insert['goodsId'],
  335. 'skuId' => $insert['skuId'],
  336. 'buyNum' => $insert['buyNum'],
  337. 'shopId' => $insert['shopId'],
  338. 'source' => $insert['source'],
  339. 'activityId' => $insert['activityId'],
  340. 'goodsCode' => $insert['goodsCode'],
  341. 'warehouseId' => $insert['warehouseId']
  342. ];//要插入的数据
  343. $insertRes = $this->objDCart->insert($insertMap);
  344. if ($insertRes === false) {
  345. $this->objDCart->rollBack();
  346. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  347. }
  348. }
  349. }
  350. $this->objDCart->commit();
  351. return ResultWrapper::success('加入成功');
  352. }
  353. /**
  354. * Doc: (des="检测商品")
  355. * User: XMing
  356. * Date: 2020/8/4
  357. * Time: 3:21 下午
  358. * @param array $goodsData
  359. * @param array $addCart
  360. * @param string $action
  361. * @return ResultWrapper
  362. */
  363. private function checkGoods(array $goodsData, array $addCart, $action = 'add')
  364. {
  365. if (empty($goodsData)) {
  366. return ResultWrapper::fail('商品信息为空', ErrorCode::$paramError);
  367. }
  368. //验证商品的状态以及库存
  369. if ($goodsData['enableStatus'] == StatusCode::$delete || $goodsData['deleteStatus'] == StatusCode::$delete) {
  370. return ResultWrapper::fail('商品已下架', ErrorCode::$paramError);
  371. }
  372. $specMultiple = $goodsData['specMultiple'];//spec数据
  373. if (empty($specMultiple)) {
  374. return ResultWrapper::fail('商品规格信息获取失败', ErrorCode::$paramError);
  375. }
  376. $specMultipleMap = [];//商品规格映射数据
  377. foreach ($specMultiple as $value) {
  378. $specMultipleMap[$value['id']] = $value;
  379. }
  380. if (!isset($specMultipleMap[$addCart['skuId']])) {
  381. return ResultWrapper::fail('规格信息不存在', ErrorCode::$paramError);
  382. }
  383. $subBuyNumResult = self::getCartBuyNumBySku($addCart['skuId'], $this->onlineUserId);
  384. if (!$subBuyNumResult->isSuccess()) {
  385. return ResultWrapper::fail($subBuyNumResult->getData(), $subBuyNumResult->getErrorCode());
  386. }
  387. $subBuyNum = floatval($subBuyNumResult->getData());
  388. //是否开启预售
  389. switch ($action){
  390. case 'add':
  391. if ($goodsData['isDistribution'] == StatusCode::$delete){
  392. if ($specMultipleMap[$addCart['skuId']]['inventory'] < $addCart['buyNum'] + $subBuyNum) {
  393. //取库存最大数
  394. $allowNum = bcsub($specMultipleMap[$addCart['skuId']]['inventory'], $subBuyNum);//最多还可以增加的数量
  395. if ($allowNum <= 0) {
  396. $addCart['buyNum'] = 0;
  397. } else {
  398. $addCart['buyNum'] = $allowNum;
  399. }
  400. //return ResultWrapper::fail('商品库存不足', ErrorCode::$paramError);
  401. }
  402. }
  403. break;
  404. case 'update':
  405. if ($goodsData['isDistribution'] == StatusCode::$delete){
  406. if ($specMultipleMap[$addCart['skuId']]['inventory'] < $addCart['buyNum']) {
  407. return ResultWrapper::success($specMultipleMap[$addCart['skuId']]['inventory']);
  408. } else {
  409. return ResultWrapper::success($addCart['buyNum']);
  410. }
  411. }else{
  412. return ResultWrapper::success($addCart['buyNum']);
  413. }
  414. break;
  415. }
  416. $addCart['isActivity'] = $specMultipleMap[$addCart['skuId']]['isActivity'];//是否是活动商品
  417. $addCart['setNum'] = $specMultipleMap[$addCart['skuId']]['setNum'];//起订数量
  418. $addCart['warehouseId'] =getArrayItem($goodsData,'warehouseId','');
  419. switch ($specMultipleMap[$addCart['skuId']]['isActivity']) {
  420. case StatusCode::$standard:
  421. $addCart['activityId'] = $specMultipleMap[$addCart['skuId']]['activity']['activityId'];
  422. $addCart['limitNum'] = $specMultipleMap[$addCart['skuId']]['activity']['limitNum'];//商品活动限购数量
  423. break;
  424. case StatusCode::$delete:
  425. $addCart['activityId'] = 0;
  426. break;
  427. }
  428. return ResultWrapper::success($addCart);
  429. }
  430. /**
  431. * app加入购物车
  432. * @param $params
  433. * @return ResultWrapper
  434. * @throws \Exception
  435. */
  436. public function addCart($params)
  437. {
  438. $this->objDCart->beginTransaction();
  439. $goodsData = $params['goodsData'];
  440. $dbResult = self::checkCart($goodsData);
  441. if (!$dbResult->isSuccess()) {
  442. return ResultWrapper::fail($dbResult->getData(), $dbResult->getErrorCode());
  443. }
  444. $checkData = $dbResult->getData();
  445. //前台验证活动商品剩余数量,和用户限购数量
  446. if ($this->isFront === true) {
  447. //将数据拆分
  448. $checkData = self::checkLimitGroup($checkData);
  449. $dbResult = self::checkLimit($checkData);
  450. if (!$dbResult->isSuccess()) {
  451. return ResultWrapper::fail($dbResult->getData(), $dbResult->getErrorCode());
  452. }
  453. $dbResult = $dbResult->getData();
  454. $mapping = $dbResult['mapping'];
  455. $checkData = $dbResult['checkData'];
  456. }
  457. unset($dbResult);
  458. $dbResult = self::existCartAndGroup($checkData);
  459. if (!$dbResult->isSuccess()) {
  460. return ResultWrapper::fail($dbResult->getData(), $dbResult->getErrorCode());
  461. }
  462. $cartData = $dbResult->getData();//分组后的数据
  463. unset($dbResult);
  464. $dbResult = true;//初始化
  465. $oldCart = $cartData['old'];//旧的数据
  466. $nowCart = $cartData['now'];//新的数据
  467. if (!empty($oldCart)) {
  468. foreach ($oldCart as $key => $val) {
  469. $sql = "UPDATE qianniao_cart_{$this->onlineEnterpriseId} SET buyNum=buyNum+{$val['buyNum']} WHERE goodsId={$val['goodsId']} AND skuId={$val['skuId']} AND userCenterId={$this->onlineUserId} AND activityId={$val['activityId']}";
  470. $dbResult = $this->objDCart->query($sql);
  471. }
  472. if ($dbResult === false) {
  473. $this->objDCart->rollBack();
  474. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  475. }
  476. }
  477. if (!empty($nowCart)) {
  478. $insert = [];
  479. foreach ($nowCart as $key => $val) {
  480. $insert[] = [
  481. 'userCenterId' => $this->onlineUserId,
  482. 'selection' => StatusCode::$standard,
  483. 'skuId' => $val['skuId'],
  484. 'goodsCode' => $val['goodsCode'],
  485. 'goodsId' => $val['goodsId'],
  486. 'shopId' => $val['shopId'],
  487. 'buyNum' => $val['buyNum'],
  488. 'source' => $val['source'],
  489. 'goodsBasicId' => $val['goodsBasicId'],
  490. 'warehouseId' => $val['warehouseId'],
  491. 'activityId' => isset($val['activityId']) ? $val['activityId'] : 0,//活动id
  492. ];
  493. }
  494. $dbResult = $this->objDCart->insert($insert, true);
  495. if ($dbResult === false) {
  496. $this->objDCart->rollBack();
  497. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  498. }
  499. }
  500. $objCustomerCache = new CustomerCache();
  501. $objCustomerCache->incrInterestCustomer($this->customerId, $this->onlineEnterpriseId);
  502. //用户限购缓存
  503. if (!empty($mapping)) {
  504. self::userLimit($mapping);
  505. }
  506. $this->objDCart->commit();
  507. return ResultWrapper::success($dbResult);
  508. }
  509. /**
  510. * 后台加入购物车
  511. * @param $params
  512. * @return ResultWrapper
  513. * @throws \Exception
  514. */
  515. public function manageAddCart($params)
  516. {
  517. $this->objDCart->beginTransaction();
  518. $goodsData = $params['goodsData'];
  519. $dbResult = self::checkCart($goodsData);
  520. if (!$dbResult->isSuccess()) {
  521. return ResultWrapper::fail($dbResult->getData(), $dbResult->getErrorCode());
  522. }
  523. $checkData = $dbResult->getData();//验证后数据
  524. $dbResult = $this->existCartAndGroup($checkData);//数据分组
  525. if (!$dbResult->isSuccess()) {
  526. return ResultWrapper::fail($dbResult->getData(), $dbResult->getErrorCode());
  527. }
  528. $cartData = $dbResult->getData();//分组后的数据
  529. unset($dbResult);
  530. $dbResult = true;//初始化
  531. $update = $cartData['old'];//旧的数据
  532. $insert = $cartData['now'];//新的数据
  533. //更新数量
  534. if (!empty($update)) {
  535. foreach ($update as $row) {
  536. $dbResult = $this->objDCart->set_inc(
  537. 'buyNum',
  538. [
  539. 'goodsId' => $row['goodsId'],
  540. 'skuId' => $row['skuId'],
  541. 'userCenterId' => $this->onlineUserId,
  542. 'activityId' => $row['activityId'],
  543. ],
  544. $row['buyNum']
  545. );
  546. if ($dbResult === false) {
  547. $this->objDCart->rollBack();
  548. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  549. }
  550. }
  551. }
  552. //insert
  553. if (!empty($insert)) {
  554. $add = [];
  555. foreach ($insert as $row) {
  556. $add[] = [
  557. 'userCenterId' => $this->onlineUserId,
  558. 'selection' => StatusCode::$standard,
  559. 'skuId' => $row['skuId'],
  560. 'goodsCode' => $row['goodsCode'],
  561. 'goodsId' => $row['goodsId'],
  562. 'shopId' => $row['shopId'],
  563. 'buyNum' => $row['buyNum'],
  564. 'source' => $row['source'],
  565. 'goodsBasicId' => $row['goodsBasicId'],
  566. 'warehouseId' => $row['warehouseId'],
  567. 'activityId' => isset($val['activityId']) ? $row['activityId'] : 0,//活动id
  568. ];
  569. }
  570. $dbResult = $this->objDCart->insert($add, true);
  571. if ($dbResult === false) {
  572. $this->objDCart->rollBack();
  573. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  574. }
  575. }
  576. $objCustomerCache = new CustomerCache();
  577. $objCustomerCache->incrInterestCustomer($this->customerId, $this->onlineEnterpriseId);
  578. $this->objDCart->commit();
  579. return ResultWrapper::success($dbResult);
  580. }
  581. /**
  582. * adj 1:+ 2:-
  583. * isUpdate true false
  584. * step 调整量
  585. * @param $checkData
  586. * @param int $adj
  587. * @param bool $isUpdate
  588. * @return ResultWrapper
  589. */
  590. private function checkLimit($checkData, $adj = 1, $isUpdate = false, $step = null)
  591. {
  592. if (empty($checkData)) return ResultWrapper::success($checkData);
  593. $mapping = [];
  594. foreach ($checkData as $key => $goods) {
  595. $skuData = $goods['specMultiple'];
  596. $skuMapping = [];
  597. foreach ($skuData as $sku) {
  598. $skuMapping[$sku['id']] = $sku;
  599. }
  600. $thisSku = $skuMapping[$goods['skuId']];//当前加入购物车规格sku详情
  601. if (
  602. $goods['isActivityPrice'] == StatusCode::$delete ||
  603. ($thisSku['isActivity'] == StatusCode::$delete && $thisSku['activityId'] == 0)) {
  604. //活动商品,已购买完,或已限购,将以原价购买
  605. continue;
  606. }
  607. $activityId = $goods['activityId'];//活动id
  608. $activityDetails = $this->objActivityLimitCache->getActivity($activityId);
  609. if (empty($activityDetails)) {
  610. return ResultWrapper::fail('活动不存在', ErrorCode::$paramError);
  611. }
  612. if ($activityDetails['startTime'] > time()) {
  613. return ResultWrapper::fail($goods['title'] . '活动尚未开始', ErrorCode::$paramError);
  614. }
  615. if ($activityDetails['endTime'] < time()) {
  616. return ResultWrapper::fail($goods['title'] . '活动已结束', ErrorCode::$paramError);
  617. }
  618. //获取活动商品剩余数量
  619. $surplusNum = $this->objActivityLimitCache->getLen($activityId, $goods['goodsId'], $goods['skuId']);
  620. if ($surplusNum == 0) {
  621. return ResultWrapper::fail($goods['title'] . '抢光了', ErrorCode::$paramError);
  622. }
  623. if ($surplusNum < $goods['buyNum']) {
  624. return ResultWrapper::fail($goods['title'] . '活动剩余数量不足', ErrorCode::$paramError);
  625. }
  626. //获取每人限购数量
  627. $userLimit = self::getLimitNum($activityDetails, $goods['goodsId'], $goods['skuId']);//用户限购数量
  628. $userSurplusNum = $this->objActivityLimitCache->getLimit($activityId, $goods['goodsId'], $goods['skuId'], $this->onlineUserId);
  629. $buyNum = 0;//初始化
  630. if ($isUpdate) {
  631. if ($adj == 1) {
  632. //加操作
  633. if ($userLimit < ($userSurplusNum + $step)) {
  634. return ResultWrapper::fail($goods['title'] . '每人限购' . $userLimit . $thisSku['unitName'], ErrorCode::$paramError);
  635. }
  636. }
  637. $buyNum = $userSurplusNum + $step;
  638. } else {
  639. //addCart
  640. if ($userLimit < ($userSurplusNum + $goods['buyNum'])) {
  641. return ResultWrapper::fail($goods['title'] . '每人限购' . $userLimit . $thisSku['unitName'], ErrorCode::$paramError);
  642. }
  643. $buyNum = $userSurplusNum + $goods['buyNum'];//
  644. }
  645. $mapping[] = [
  646. 'activityId' => $activityId,
  647. 'goodsId' => $goods['goodsId'],
  648. 'skuId' => $goods['skuId'],
  649. 'buyNum' => $buyNum,
  650. ];
  651. }
  652. $data = [
  653. 'checkData' => $checkData,
  654. 'mapping' => $mapping,
  655. ];
  656. return ResultWrapper::success($data);
  657. }
  658. /**
  659. * 缓存用户限购
  660. * @param $data
  661. * @return mixed
  662. */
  663. private function userLimit($data)
  664. {
  665. $result = false;
  666. foreach ($data as $goods) {
  667. $result = $this->objActivityLimitCache->writeLimit($goods['activityId'], $goods['goodsId'], $goods['skuId'], $this->onlineUserId, $goods['buyNum']);
  668. }
  669. return $result;
  670. }
  671. /**
  672. * 获取商品的限购数量
  673. * @param $data
  674. * @param $goodsId
  675. * @param $skuId
  676. * @return int|mixed
  677. */
  678. private function getLimitNum($data, $goodsId, $skuId)
  679. {
  680. if (empty($data)) return $data;
  681. $goodsData = $data['activityGoods'];
  682. $mapping = [];
  683. foreach ($goodsData as $goods) {
  684. $mapping[$goods['goodsId'] . $goods['skuId']] = $goods;
  685. }
  686. return isset($mapping[$goodsId . $skuId]['limitNum']) ? $mapping[$goodsId . $skuId]['limitNum'] : 0;
  687. }
  688. /**
  689. * 检测购物车商品(最新的方法)
  690. * @param $data
  691. * @return ResultWrapper
  692. * @throws \Exception
  693. */
  694. public function checkCart($data)
  695. {
  696. if (empty($data)) return ResultWrapper::fail('购物车数据为空', ErrorCode::$paramError);
  697. $objMGoods = new MGoods($this->onlineEnterpriseId, $this->isFront, $this->onlineUserId);
  698. //验证商品数据
  699. foreach ($data as $key => $val) {
  700. //获取商品详情
  701. $details = $objMGoods->getGoodsInfo($val['goodsId']);
  702. if (!$details->isSuccess()) {
  703. return ResultWrapper::fail($details->getData(), $details->getErrorCode());
  704. }
  705. $goods = $details->getData();//商品数据
  706. if (empty($details)) {
  707. return ResultWrapper::fail($val['goodsCode'] . '查询商品失败', ErrorCode::$paramError);
  708. }
  709. //验证商品状态
  710. if ($goods['enableStatus'] == StatusCode::$delete || $goods['deleteStatus'] == StatusCode::$delete) {
  711. return ResultWrapper::fail($goods['title'] . ' 已下架', ErrorCode::$paramError);
  712. }
  713. $specMultiple = $goods['specMultiple'];//spec数据
  714. $specMapping = [];//Map
  715. foreach ($specMultiple as $spec) {
  716. $specMapping[$spec['id']] = $spec;
  717. }
  718. if (!isset($specMapping[$val['skuId']])) {
  719. Logger::logs(E_USER_ERROR, '规格获取失败', __CLASS__, __LINE__, $specMapping);
  720. return ResultWrapper::fail($goods['title'] . '规格获取失败', ErrorCode::$paramError);
  721. }
  722. //验证商品库存
  723. if ($this->preSale == StatusCode::$delete) {
  724. if ($specMapping[$val['skuId']]['inventory'] < $val['buyNum']) {
  725. $specName = self::createSpecName($specMapping[$val['skuId']]);
  726. return ResultWrapper::fail($goods['title'] . $specName . ' 库存不足', ErrorCode::$paramError);
  727. }
  728. }
  729. $data[$key]['basicGoodsId'] = $val['goodsBasicId'];//为了兼容之前的代码
  730. $data[$key]['skuData'] = [];//兼容之前数据
  731. $data[$key]['specMultiple'] = $specMultiple;//spec
  732. $data[$key]['warehouseId'] = $goods['warehouseId'];//店铺所在仓库
  733. $data[$key]['title'] = $goods['title'];//商品名称
  734. $data[$key]['setNum'] = $specMapping[$val['skuId']]['setNum'];//加入购物车起订量
  735. $data[$key]['activityId'] = 0;//初始化活动id
  736. //todo(优化秒杀商品活动)
  737. if ($specMapping[$val['skuId']]['isActivity'] == StatusCode::$standard) {
  738. //商品存在活动时,没有起订量
  739. $data[$key]['setNum'] = 1;
  740. }
  741. $data[$key]['inventory'] = $specMapping[$val['skuId']]['inventory'];//加入购物车商品规格的可用库存
  742. $data[$key]['isActivityPrice'] = $specMapping[$val['skuId']]['isActivityPrice'];
  743. if ($specMapping[$val['skuId']]['isActivityPrice'] == StatusCode::$standard) {
  744. $data[$key]['activityId'] = $specMapping[$val['skuId']]['activity']['activityId'];
  745. }
  746. $data[$key]['limitNum'] = $specMapping[$val['skuId']]['limitNum'];//商品限购数量
  747. }
  748. return ResultWrapper::success($data);
  749. }
  750. /**
  751. * 生成属性名
  752. * @param $data
  753. * @return mixed|string
  754. */
  755. public static function createSpecName($data)
  756. {
  757. if (empty($data)) {
  758. return '';
  759. }
  760. $unitName = $data['unitName'];
  761. if (!empty($data['specGroup'])) {
  762. $arrSpecName = array_column($data['specGroup'], 'specValueName');
  763. $specName = rtrim(implode('_', $arrSpecName), '_');
  764. $unitName .= '_' . $specName;
  765. }
  766. return $unitName;
  767. }
  768. /**
  769. * 将加入购物车的数据进行分组[old=>在购物车中已存在,now=>新加入购物车商品]
  770. * @param $nowData [加入购物车的数据]
  771. * @return ResultWrapper
  772. */
  773. private function existCartAndGroup($nowData)
  774. {
  775. // 查询当前购物车所有商品数据
  776. $dbResult = self::getCartByUid();
  777. if (!$dbResult->isSuccess()) {
  778. return ResultWrapper::fail('获取购物车数据失败', ErrorCode::$dberror);
  779. }
  780. $oldCartData = $dbResult->getData();
  781. // 对所有商品数据生成hash值,用于判断新加入的商品是否已经存在于购物车之中了
  782. $allHash = [];
  783. foreach ($oldCartData as $key => $val) {
  784. $allHash[] = md5($val['goodsId'] . $val['shopId'] . $val['skuId'] . $val['activityId']);
  785. }
  786. $old = [];//旧的数据
  787. $now = [];//新的数据
  788. foreach ($nowData as $key => &$val) {
  789. $md5 = md5($val['goodsId'] . $val['shopId'] . $val['skuId'] . $val['activityId']);
  790. if (in_array($md5, $allHash)) {
  791. $old[] = $val;
  792. } else {
  793. //之前购物车中没有此商品,检查起订量
  794. if ($val['setNum'] != 0) {
  795. //具有起订量
  796. if ($val['buyNum'] < $val['setNum']) {
  797. $val['buyNum'] = $val['setNum'];
  798. }
  799. }
  800. $now[] = $val;
  801. }
  802. }
  803. $groupData = [
  804. 'old' => $old,
  805. 'now' => $now
  806. ];
  807. return ResultWrapper::success($groupData);
  808. }
  809. /**
  810. * Doc: (des="更新购车商品数量")
  811. * User: XMing
  812. * Date: 2020/8/6
  813. * Time: 9:43 上午
  814. * @param array $params
  815. * @return ResultWrapper
  816. * @throws \Exception
  817. */
  818. public function updateBuyNumApi(array $params)
  819. {
  820. // 如果数量改成0则删除
  821. if (bccomp($params['buyNum'], 0, 0) === 0) {
  822. $dbResult = $this->objDCart->delete(['id' => $params['cartId']]);
  823. if ($dbResult === false) {
  824. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  825. }
  826. return ResultWrapper::success('操作成功');
  827. }
  828. $cart = $this->objDCart->get($params['cartId']);
  829. if ($cart === false) {
  830. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  831. }
  832. if (empty($cart)) {
  833. return ResultWrapper::success(true);
  834. }
  835. $goodsInfoResult = $this->objMGoods->getGoodsInfo($cart['goodsId']);
  836. if (!$goodsInfoResult->isSuccess()) {
  837. return ResultWrapper::fail($goodsInfoResult->getData(), $goodsInfoResult->getErrorCode());
  838. }
  839. $goodsInfo = $goodsInfoResult->getData();
  840. $checkResult = self::checkGoods($goodsInfo, [
  841. 'skuId' => $cart['skuId'],
  842. 'buyNum' => $params['buyNum'],
  843. ], 'update');
  844. unset($data);
  845. if (!$checkResult->isSuccess()) {
  846. return ResultWrapper::fail($checkResult->getData(), $checkResult->getErrorCode());
  847. }
  848. $allBuyNum = $checkResult->getData();
  849. $params['buyNum'] = $allBuyNum;
  850. $dbResult = $this->objDCart->update(['buyNum' => $params['buyNum']], $params['cartId']);
  851. if ($dbResult === false) {
  852. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  853. }
  854. return ResultWrapper::success('操作成功');
  855. }
  856. /**
  857. * 修改购物车商品购买数量
  858. * @param $params
  859. * @return ResultWrapper
  860. * @throws \Exception
  861. */
  862. public function updateBuyNum($params)
  863. {
  864. if (bccomp($params['buyNum'], 0, 0) === 0) {
  865. $dbResult = $this->objDCart->delete(['id' => $params['cartId']]);
  866. if ($dbResult === false) {
  867. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  868. }
  869. return ResultWrapper::success('操作成功');
  870. } else {
  871. $goods = self::getCartInventory(['id' => $params['cartId']]);
  872. if (!$goods->isSuccess()) {
  873. return ResultWrapper::fail($goods->getData(), ErrorCode::$dberror);
  874. }
  875. $goods = $goods->getData();
  876. $inventoryNum = isset($goods[0]['inventory']) ? $goods[0]['inventory'] : 0;//商品剩余库存
  877. $setNum = isset($goods[0]['setNum']) ? $goods[0]['setNum'] : 0;//商品规格起订数量
  878. $title = isset($goods[0]['title']) ? $goods[0]['title'] : '';//商品名称
  879. if (bccomp($inventoryNum, $params['buyNum'], 8) === -1) {
  880. return ResultWrapper::fail('商品库存不足', ErrorCode::$paramError);
  881. }
  882. if (bccomp($params['buyNum'], $setNum, 0) === -1) {
  883. return ResultWrapper::fail($title . '最小起订数量' . $setNum, ErrorCode::$paramError);
  884. }
  885. if ($this->isFront) {
  886. $dbResult = $this->objDCart->get(['id' => $params['cartId']]);
  887. if ($dbResult === false) {
  888. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  889. }
  890. $step = bcsub($params['buyNum'], $dbResult['buyNum'], 0);//调整量
  891. //所有减操作正常进行
  892. if (bccomp($params['buyNum'], $dbResult['buyNum']) === 0) {
  893. return ResultWrapper::success('操作成功');
  894. } elseif (bccomp($params['buyNum'], $dbResult['buyNum'], 0) === -1) {
  895. $adj = 2;//减操作
  896. $result = $this->objDCart->update(['buyNum' => $params['buyNum']], ['id' => $params['cartId']]);
  897. if ($result === false) {
  898. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  899. }
  900. return ResultWrapper::success('操作成功');
  901. } elseif (bccomp($params['buyNum'], $dbResult['buyNum'], 0) === 1) {
  902. $adj = 1;//加操作
  903. $insert = [
  904. 'goodsId' => $dbResult['goodsId'],
  905. 'goodsBasicId' => $dbResult['goodsBasicId'],
  906. 'shopId' => $dbResult['shopId'],
  907. 'source' => $dbResult['source'],
  908. 'skuId' => $dbResult['skuId'],
  909. 'buyNum' => $step,
  910. 'goodsCode' => $dbResult['goodsCode']
  911. ];
  912. $result = self::addCart(['goodsData' => [$insert]]);
  913. if (!$result->isSuccess()) {
  914. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  915. }
  916. return ResultWrapper::success('操作成功');
  917. }
  918. } else {
  919. $dbResult = $this->objDCart->update(['buyNum' => $params['buyNum']], $params['cartId']);
  920. if ($dbResult === false) {
  921. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  922. }
  923. return ResultWrapper::success('操作成功');
  924. }
  925. }
  926. }
  927. /**
  928. * Doc: (des="获取用户的购物车数据")
  929. * User: XMing
  930. * Date: 2020/8/5
  931. * Time: 3:15 下午
  932. * @param integer $isZero
  933. * @param boolean $isCashier
  934. * @return ResultWrapper
  935. * @throws \Exception
  936. */
  937. public function getCartByUserCenterIdApi(): ResultWrapper
  938. {
  939. // 查询购物车数据
  940. $sql = 'SELECT id,goodsId,goodsCode,buyNum,shopId,source,goodsBasicId,selection,skuId,warehouseId,activityId,cashierUid FROM qianniao_cart_'.$this->onlineEnterpriseId.' WHERE userCenterId = '.$this->onlineUserId.' FOR UPDATE';
  941. $cartList = $this->objDCart->query($sql);
  942. if ($cartList === false){
  943. return ResultWrapper::fail($this->objDCart->error(),ErrorCode::$dberror);
  944. }
  945. if (empty($cartList)) {
  946. return ResultWrapper::success(['data' => [], 'total' => 0]);
  947. }
  948. // 提取活动ids
  949. $activityIds = [];
  950. foreach ($cartList as $item){
  951. if(!empty($item['activityId']) && !in_array($item['activityId'],$activityIds)){
  952. $activityIds[] = $item['activityId'];
  953. }
  954. }
  955. if (!empty($activityIds)){
  956. $groupResult = self::cartGroupData($cartList);
  957. if (!$groupResult->isSuccess()) {
  958. return ResultWrapper::fail($groupResult->getData(), $groupResult->getErrorCode());
  959. }
  960. $cartList = $groupResult->getData();
  961. if (empty($cartList)) {
  962. return ResultWrapper::success(['data' => [], 'total' => 0]);
  963. }
  964. }
  965. $dataResult = self::formatGoodsAndShop($cartList);
  966. if (!$dataResult->isSuccess()) {
  967. return ResultWrapper::fail($dataResult->getData(), $dataResult->getErrorCode());
  968. }
  969. $data = $dataResult->getData();
  970. $return = [
  971. 'data' => $data,
  972. 'total' => empty($data) ? 0 : count($data),
  973. ];
  974. return ResultWrapper::success($return);
  975. }
  976. /**
  977. * Doc: (des="抹零操作")
  978. * User: XMing
  979. * Date: 2020/9/12
  980. * Time: 7:08 下午
  981. * @param array $data
  982. * @param int $isZero
  983. * @return ResultWrapper
  984. */
  985. private function buildZero($data, $isZero)
  986. {
  987. //是否开启抹零
  988. $data['rem_money'] = '0.00';
  989. $settingResult = $this->objMCashierSettings->get();
  990. if (!$settingResult->isSuccess()) {
  991. return ResultWrapper::fail($settingResult->getData(), $settingResult->getErrorCode());
  992. }
  993. $setting = $settingResult->getData();
  994. if (!is_array($setting)) {
  995. //不是数组时,说明未配置
  996. return ResultWrapper::success($data);
  997. }
  998. if (!isset($setting['add_form'])) {
  999. return ResultWrapper::fail('收银台配置错误', ErrorCode::$paramError);
  1000. }
  1001. if (!isset($setting['add_form']['zero_set'])) {
  1002. return ResultWrapper::fail('收银台参数配置错误', ErrorCode::$paramError);
  1003. }
  1004. $zeroSet = $setting['add_form']['zero_set'];
  1005. if ($zeroSet['status'] == StatusCode::$delete) {
  1006. //没有开启抹零,前台需要在抹零那判断是否开启
  1007. return ResultWrapper::success($data);
  1008. }
  1009. $payMoney = $data['payMoney'];
  1010. if ($zeroSet['auto_zero'] == StatusCode::$standard) {
  1011. //开启了自动抹零
  1012. switch ($zeroSet['type']) {
  1013. case 1:
  1014. //分
  1015. $newPatMoney = bcadd($payMoney, 0, 1);
  1016. break;
  1017. case 2:
  1018. //角
  1019. $newPatMoney = bcadd($payMoney, 0, 0);
  1020. break;
  1021. case 3:
  1022. //到角
  1023. $newPatMoney = round($payMoney, 1);
  1024. break;
  1025. case 4:
  1026. //到元
  1027. $newPatMoney = round($payMoney, 0);
  1028. break;
  1029. default:
  1030. $newPatMoney = 0;
  1031. break;
  1032. }
  1033. $rem_money = bcsub($payMoney, $newPatMoney, 2);
  1034. $data['payMoney'] = $newPatMoney;
  1035. $data['rem_money'] = $rem_money;
  1036. } else {
  1037. if ($isZero == StatusCode::$standard) {
  1038. //手动抹零
  1039. switch ($zeroSet['type']) {
  1040. case 1:
  1041. //分
  1042. $newPatMoney = bcadd($payMoney, 0, 1);
  1043. break;
  1044. case 2:
  1045. //角
  1046. $newPatMoney = bcadd($payMoney, 0, 0);
  1047. break;
  1048. case 3:
  1049. //到角
  1050. $newPatMoney = round($payMoney, 1);
  1051. break;
  1052. case 4:
  1053. //到元
  1054. $newPatMoney = round($payMoney, 0);
  1055. break;
  1056. default:
  1057. $newPatMoney = 0;
  1058. break;
  1059. }
  1060. $rem_money = bcsub($payMoney, $newPatMoney, 2);
  1061. $data['payMoney'] = $newPatMoney;
  1062. $data['rem_money'] = $rem_money;
  1063. }
  1064. }
  1065. return ResultWrapper::success($data);
  1066. }
  1067. /**
  1068. * Doc: (des="将购物车的数据进行分组 活动商品的限购级别以及价格,全部都基于此方法组装的数据")
  1069. * User: XMing
  1070. * Date: 2020/8/5
  1071. * Time: 3:21 下午
  1072. * @param array $data
  1073. * @return ResultWrapper
  1074. */
  1075. private function cartGroupData(array $data)
  1076. {
  1077. if (empty($data)) {
  1078. return ResultWrapper::success([]);
  1079. }
  1080. // 获取限购级别设置项
  1081. $BasicSetupResult = $this->objMBasicSetup->getBasicField('limitLevel');
  1082. if (!$BasicSetupResult->isSuccess()) {
  1083. return ResultWrapper::fail($BasicSetupResult->getData(), $BasicSetupResult->getErrorCode());
  1084. }
  1085. $BasicSetup = $BasicSetupResult->getData();//之前的企业可鞥未设置此字段,给默认值5
  1086. $limitLevel = isset($BasicSetup['limitLevel']) ? $BasicSetup['limitLevel'] : StatusCode::$standard;//限购级别
  1087. $selectStateMap = [];
  1088. //将同sku合并到一起s
  1089. $dataMap = [];
  1090. $skuBuyMap = [];
  1091. foreach ($data as $datum) {
  1092. $selectStateMap[$datum['skuId'] . ':' . $datum['activityId']] = $datum['selection'];
  1093. $dataMap[$datum['skuId']] = [
  1094. 'id' => $datum['id'],
  1095. 'goodsId' => $datum['goodsId'],
  1096. 'goodsCode' => $datum['goodsCode'],
  1097. 'shopId' => $datum['shopId'],
  1098. 'source' => $datum['source'],
  1099. 'goodsBasicId' => $datum['goodsBasicId'],
  1100. 'selection' => $datum['selection'],
  1101. 'skuId' => $datum['skuId'],
  1102. 'warehouseId' => $datum['warehouseId'],
  1103. 'activityId' => 0,
  1104. ];
  1105. $skuBuyMap[$datum['skuId']][] = $datum['buyNum'];
  1106. }
  1107. $this->selectStateMap = $selectStateMap;
  1108. foreach ($dataMap as $skuId => $value) {
  1109. $dataMap[$skuId]['buyNum'] = array_sum($skuBuyMap[$skuId]);
  1110. }
  1111. $data = array_values($dataMap);
  1112. //获取客户信息
  1113. $customer = $this->objDCustomer->get(['userCenterId' => $this->onlineUserId], 'id,type');
  1114. if ($customer === false) {
  1115. return ResultWrapper::fail($this->objDCustomer->error(), ErrorCode::$dberror);
  1116. }
  1117. if (empty($customer)) {
  1118. $customer = [];
  1119. }
  1120. //获取商品是否是活动商品
  1121. $allSkuIds = [];
  1122. foreach ($data as $datum) {
  1123. $allSkuIds[] = $datum['skuId'];
  1124. }
  1125. $activityResult = $this->objMActivity->getActivity([
  1126. 'skuIds' => implode(',', $allSkuIds),
  1127. 'customerType' => isset($customer['type']) ? $customer['type'] : 0,
  1128. ]);
  1129. if (!$activityResult->isSuccess()) {
  1130. return ResultWrapper::fail($activityResult->getData(), $activityResult->getErrorCode());
  1131. }
  1132. $activity = $activityResult->getData();
  1133. $activityMap = [];
  1134. foreach ($activity as $item) {
  1135. $activityMap[$item['skuId']] = $item;
  1136. }
  1137. $update = [];
  1138. foreach ($data as $item) {
  1139. if (!isset($activityMap[$item['skuId']])) {
  1140. //没有活动不用处理
  1141. $update[] = self::buildData($item, $item['buyNum'], 0, $this->onlineUserId);
  1142. continue;
  1143. }
  1144. //是活动商品
  1145. //1。活动商品数量是否足够
  1146. $len = $this->objActivityLimitCache->getLen($activityMap[$item['skuId']]['activityId'], $item['goodsId'], $item['skuId']);
  1147. if ($len == 0) {
  1148. //活动商品已购买完
  1149. $update[] = self::buildData($item, $item['buyNum'], 0, $this->onlineUserId);
  1150. continue;
  1151. }
  1152. //2。剩余数量
  1153. //2.1 剩余数量足够
  1154. if ($len >= $item['buyNum']) {
  1155. //获取用户已购数量
  1156. $userLimit = $this->objActivityLimitCache->getLimit($activityMap[$item['skuId']]['activityId'], $item['goodsId'], $item['skuId'], $this->onlineUserId);
  1157. //3。用户限购
  1158. switch ($limitLevel) {
  1159. case StatusCode::$standard:
  1160. if ($activityMap[$item['skuId']]['limitNum'] == 0) {
  1161. $update[] = self::buildData($item, $item['buyNum'], 0, $this->onlineUserId);
  1162. continue 2;
  1163. }
  1164. //3。1 活动期间限购
  1165. if ($userLimit > $activityMap[$item['skuId']]['limitNum']) {
  1166. //已经购买过活动商品了,剩下的都恢复原价
  1167. $update[] = self::buildData($item, $item['buyNum'], 0, $this->onlineUserId);
  1168. continue 2;
  1169. }
  1170. $allowBuyNum = $activityMap[$item['skuId']]['limitNum'] - $userLimit;
  1171. if ($allowBuyNum == 0) {
  1172. $update[] = self::buildData($item, $item['buyNum'], 0, $this->onlineUserId);
  1173. continue 2;
  1174. }
  1175. if ($item['buyNum'] <= $allowBuyNum) {
  1176. $update[] = self::buildData($item, $item['buyNum'], $activityMap[$item['skuId']]['activityId'], $this->onlineUserId);
  1177. continue 2;
  1178. }
  1179. if ($item['buyNum'] > $allowBuyNum) {
  1180. //分解
  1181. $update[] = self::buildData($item, $allowBuyNum, $activityMap[$item['skuId']]['activityId'], $this->onlineUserId);
  1182. $update[] = self::buildData($item, $item['buyNum'] - $allowBuyNum, 0, $this->onlineUserId);//原价不参与活动
  1183. continue 2;
  1184. }
  1185. break;
  1186. case StatusCode::$delete:
  1187. //3。2 单笔订单限购买
  1188. if ($activityMap[$item['skuId']]['limitNum'] < $item['buyNum']) {
  1189. //购车中的商品大于单笔限购,分解
  1190. $update[] = self::buildData($item, $activityMap[$item['skuId']]['limitNum'], $activityMap[$item['skuId']]['activityId'], $this->onlineUserId);
  1191. $update[] = self::buildData($item, $item['buyNum'] - $activityMap[$item['skuId']]['limitNum'], 0, $this->onlineUserId);
  1192. continue 2;
  1193. }
  1194. if ($activityMap[$item['skuId']]['limitNum'] >= $item['buyNum']) {
  1195. //都是活动价格
  1196. $update[] = self::buildData($item, $item['buyNum'], $activityMap[$item['skuId']]['activityId'], $this->onlineUserId);
  1197. continue 2;
  1198. }
  1199. break;
  1200. }
  1201. } else {
  1202. //2。2 剩余数量不足
  1203. $update[] = self::buildData($item, $item['buyNum'], 0, $this->onlineUserId);
  1204. continue;
  1205. }
  1206. }
  1207. if (empty($update)) {
  1208. return ResultWrapper::fail('获取购物车数据失败', ErrorCode::$paramError);
  1209. }
  1210. $this->objDCart->beginTransaction();
  1211. $deleteRes = $this->objDCart->delete(['userCenterId' => $this->onlineUserId]);
  1212. if ($deleteRes === false) {
  1213. $this->objDCart->rollBack();
  1214. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1215. }
  1216. $insertRes = $this->objDCart->insert($update, true);
  1217. if ($insertRes === false) {
  1218. $this->objDCart->rollBack();
  1219. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1220. }
  1221. $this->objDCart->commit();
  1222. $where['userCenterId'] = $this->onlineUserId;
  1223. $cartList = $this->objDCart->select($where);
  1224. if ($cartList === false) {
  1225. $this->objDCart->rollBack();
  1226. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1227. }
  1228. return ResultWrapper::success($cartList);
  1229. }
  1230. /**
  1231. * Doc: (des="")
  1232. * User: XMing
  1233. * Date: 2020/8/5
  1234. * Time: 5:22 下午
  1235. * @param array $item
  1236. * @param int $buyNum
  1237. * @param int $activityId
  1238. * @param int $userCenterId
  1239. * @return array
  1240. */
  1241. private function buildData(array $item, int $buyNum, int $activityId, int $userCenterId)
  1242. {
  1243. $data = [
  1244. 'goodsId' => $item['goodsId'],
  1245. 'goodsCode' => $item['goodsCode'],
  1246. 'shopId' => $item['shopId'],
  1247. 'source' => $item['source'],
  1248. 'goodsBasicId' => $item['goodsBasicId'],
  1249. 'selection' => isset($this->selectStateMap[$item['skuId'] . ':' . $activityId]) ? $this->selectStateMap[$item['skuId'] . ':' . $activityId] : StatusCode::$delete,
  1250. 'skuId' => $item['skuId'],
  1251. 'warehouseId' => $item['warehouseId'],
  1252. 'activityId' => $activityId,
  1253. 'buyNum' => $buyNum,
  1254. 'userCenterId' => $userCenterId
  1255. ];
  1256. return $data;
  1257. }
  1258. /**
  1259. * app获取购物车内的数据
  1260. * @return ResultWrapper
  1261. * @throws \Exception
  1262. */
  1263. public function getCartByUserCenterId()
  1264. {
  1265. $result = $this->objDCart->select(['userCenterId' => $this->onlineUserId], 'id,goodsId,goodsCode,buyNum,shopId,source,goodsBasicId,selection,skuId,warehouseId,activityId', 'createTime DESC');
  1266. if ($result === false) {
  1267. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1268. }
  1269. $result = self::formatMerge($result);
  1270. if (!$result->isSuccess()) {
  1271. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  1272. }
  1273. $dataResult = self::formatGoodsAndShop($result->getData());
  1274. if (!$dataResult->isSuccess()) {
  1275. return ResultWrapper::fail($dataResult->getData(), $dataResult->getErrorCode());
  1276. }
  1277. $data = $dataResult->getData();
  1278. $return = [
  1279. 'data' => $data,
  1280. 'total' => empty($data) ? 0 : count($data),
  1281. ];
  1282. return ResultWrapper::success($return);
  1283. }
  1284. /**
  1285. * 后台获取购车数据
  1286. * @return ResultWrapper
  1287. * @throws \Exception
  1288. */
  1289. public function getManageCartByUserCenterId()
  1290. {
  1291. $result = $this->objDCart->select(['userCenterId' => $this->onlineUserId], 'id,goodsId,goodsCode,buyNum,shopId,goodsBasicId,selection,skuId,warehouseId,activityId', 'createTime DESC');
  1292. if ($result === false) {
  1293. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1294. }
  1295. $dataResult = self::formatGoodsAndShop($result);
  1296. if (!$dataResult->isSuccess()) {
  1297. return ResultWrapper::fail($dataResult->getData(), $dataResult->getErrorCode());
  1298. }
  1299. $data = $dataResult->getData();
  1300. $return = [
  1301. 'data' => $data,
  1302. 'total' => empty($data) ? 0 : count($data),
  1303. ];
  1304. return ResultWrapper::success($return);
  1305. }
  1306. /**
  1307. * 获取当前用户的购物车数据
  1308. */
  1309. private function getCartByUid()
  1310. {
  1311. $result = $this->objDCart->select(['userCenterId' => $this->onlineUserId], 'id,goodsId,goodsCode,buyNum,shopId,goodsBasicId,selection,skuId,warehouseId,activityId,cashierUid', 'createTime DESC');
  1312. if ($result === false) {
  1313. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1314. }
  1315. return ResultWrapper::success($result);
  1316. }
  1317. /**
  1318. * 更新选中状态
  1319. * @param $params
  1320. * @return ResultWrapper
  1321. * @throws \Exception
  1322. */
  1323. public function updateSelection($params)
  1324. {
  1325. $dbResult = false;
  1326. switch ($params['type']) {
  1327. case self::$type['single']:
  1328. if (empty($params['id'])) {
  1329. return ResultWrapper::fail('缺少cartId', ErrorCode::$paramError);
  1330. }
  1331. $dbResult = $this->objDCart->update(['selection' => $params['selection']], $params['id']);
  1332. break;
  1333. case self::$type['shop']:
  1334. if (empty($params['shopId'])) {
  1335. return ResultWrapper::fail('缺少shopId', ErrorCode::$paramError);
  1336. }
  1337. $dbResult = $this->objDCart->update(['selection' => $params['selection']], ['shopId' => $params['shopId'], 'userCenterId' => $this->onlineUserId]);
  1338. break;
  1339. case self::$type['all']:
  1340. $dbResult = $this->objDCart->update(['selection' => $params['selection']], ['userCenterId' => $this->onlineUserId]);
  1341. break;
  1342. }
  1343. if ($dbResult === false) {
  1344. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1345. }
  1346. return ResultWrapper::success($dbResult);
  1347. }
  1348. /**
  1349. * 删除购物车中商品(可批量)
  1350. * @param $ids
  1351. * @param boolean $order 是否是来自订单
  1352. * @return ResultWrapper
  1353. */
  1354. public function delCart($ids, $order = false)
  1355. {
  1356. $dbResult = $this->objDCart->select($ids, 'id,goodsId,buyNum,skuId,activityId');
  1357. if ($dbResult === false) {
  1358. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1359. }
  1360. //限购减对应数据
  1361. if ($order === true) {
  1362. foreach ($dbResult as $goods) {
  1363. if ($goods['activityId'] != 0) {
  1364. $userSurplusNum = $this->objActivityLimitCache->getLimit($goods['activityId'], $goods['goodsId'], $goods['skuId'], $this->onlineUserId);
  1365. $mapping[] = [
  1366. 'activityId' => $goods['activityId'],
  1367. 'goodsId' => $goods['goodsId'],
  1368. 'skuId' => $goods['skuId'],
  1369. 'buyNum' => $userSurplusNum + $goods['buyNum'],
  1370. ];
  1371. self::userLimit($mapping);
  1372. }
  1373. }
  1374. }
  1375. $dbResult = $this->objDCart->delete($ids);
  1376. if ($dbResult === false) {
  1377. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1378. }
  1379. return ResultWrapper::success($dbResult);
  1380. }
  1381. /**
  1382. * Doc: (des="判断字符串是否是一个整数")
  1383. * User: XMing
  1384. * Date: 2020/10/12
  1385. * Time: 10:29 上午
  1386. * @param $num
  1387. * @return bool
  1388. */
  1389. private static function isInteger($num)
  1390. {
  1391. if ($num - floor($num) != 0) {
  1392. return false;
  1393. }
  1394. return true;
  1395. }
  1396. /**
  1397. * 拼装购物车数据(商铺->商品->规格->库存->价格->分组)
  1398. * 此方法正在修改,勿动
  1399. * @param $data
  1400. * @return ResultWrapper
  1401. * @throws \Exception
  1402. */
  1403. private function formatGoodsAndShop($data, $userCouponId = null, $vipCardId = null)
  1404. {
  1405. foreach ($data as &$value) {
  1406. $value['buyNum'] = self::isInteger($value['buyNum']) ? (int)$value['buyNum'] : $value['buyNum'];
  1407. }
  1408. unset($value);
  1409. $result = self::formatShop($data);//格式化店铺数据
  1410. $result = self::formatGoods($result);//格式化商品数据
  1411. if ($this->isFront === true) {
  1412. $result = self::calExpress($result);
  1413. if (!$result->isSuccess()) {
  1414. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  1415. }
  1416. $result = $result->getData();
  1417. }
  1418. // 价格处理
  1419. $priceResult = self::formatPrice($result);
  1420. if (!$priceResult->isSuccess()) {
  1421. return ResultWrapper::fail($priceResult->getData(), $priceResult->getErrorCode());
  1422. }
  1423. $result = $priceResult->getData();
  1424. $inventoryResult = self::formatSkuInventory($result);//处理库存
  1425. if (!$inventoryResult->isSuccess()){
  1426. return ResultWrapper::fail($inventoryResult->getData(),$inventoryResult->getErrorCode());
  1427. }
  1428. $result = $inventoryResult->getData();
  1429. $result = self::formatGroup($result, $userCouponId, $vipCardId);
  1430. if(!$result->isSuccess()){
  1431. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  1432. }
  1433. return ResultWrapper::success($result->getData());
  1434. }
  1435. /**
  1436. * @param $data
  1437. * @return ResultWrapper
  1438. */
  1439. private function formatMerge($data)
  1440. {
  1441. if (empty($data)) return ResultWrapper::success($data);
  1442. foreach ($data as $goods) {
  1443. if ($goods['activityId'] == 0) {
  1444. continue;
  1445. }
  1446. //获取活动详情
  1447. $activity = $this->objActivityLimitCache->getActivity($goods['activityId']);
  1448. if ($activity['enableStatus'] == StatusCode::$delete
  1449. || time() < $activity['startTime']
  1450. || time() > $activity['endTime']) {
  1451. //活动被下架,将此条数据合并到普通商品
  1452. $cartResult = $this->objDCart->get(['goodsId' => $goods['goodsId'], 'shopId' => $goods['shopId'], 'skuId' => $goods['skuId'], 'activityId' => 0, 'userCenterId' => $this->onlineUserId]);
  1453. if (empty($cartResult)) {
  1454. //addCart
  1455. $insert = [
  1456. 'userCenterId' => $this->onlineUserId,
  1457. 'selection' => StatusCode::$standard,
  1458. 'skuId' => $goods['skuId'],
  1459. 'goodsCode' => $goods['goodsCode'],
  1460. 'goodsId' => $goods['goodsId'],
  1461. 'shopId' => $goods['shopId'],
  1462. 'buyNum' => $goods['buyNum'],//购买数量
  1463. 'source' => $goods['source'],
  1464. 'goodsBasicId' => $goods['goodsBasicId'],
  1465. 'warehouseId' => $goods['warehouseId'],
  1466. 'activityId' => 0,//活动id
  1467. ];
  1468. $dbResult = $this->objDCart->insert($insert);
  1469. } else {
  1470. //update
  1471. $dbResult = $this->objDCart->set_inc('buyNum', ['id' => $cartResult['id']], $goods['buyNum']);
  1472. }
  1473. $delResult = $this->objDCart->delete(['id' => $goods['id']]);
  1474. //获取用户限购数量
  1475. $userLimit = $this->objActivityLimitCache->getLimit($goods['activityId'], $goods['goodsId'], $goods['skuId'], $this->onlineUserId);
  1476. $this->objActivityLimitCache->writeLimit($goods['activityId'], $goods['goodsId'], $goods['skuId'], $this->onlineUserId, $userLimit - $goods['buyNum']);
  1477. if ($delResult === false) {
  1478. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1479. }
  1480. if ($dbResult === false) {
  1481. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1482. }
  1483. }
  1484. }
  1485. $result = $this->objDCart->select(['userCenterId' => $this->onlineUserId], 'id,goodsId,goodsCode,buyNum,shopId,goodsBasicId,selection,skuId,warehouseId,activityId', 'createTime DESC');
  1486. if ($result === false) {
  1487. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  1488. }
  1489. return ResultWrapper::success($result);
  1490. }
  1491. /**
  1492. * 获取店铺信息
  1493. * (原数据上+店铺名称+店铺logo)
  1494. * @param $data
  1495. * @return mixed
  1496. * @throws \Exception
  1497. */
  1498. private function formatShop($data)
  1499. {
  1500. if (empty($data)) return $data;
  1501. $objMGoods = new MShop($this->onlineEnterpriseId, $this->onlineUserId);
  1502. $shopData = $objMGoods->getShopName(array_unique(array_column($data, 'shopId')));
  1503. if (!empty($shopData)) {
  1504. foreach ($data as &$shop) {
  1505. $shop['shopName'] = isset($shopData[$shop['shopId']]['name']) ? $shopData[$shop['shopId']]['name'] : '';
  1506. $shop['shopLogo'] = isset($shopData[$shop['shopId']]['logo']) ? $shopData[$shop['shopId']]['logo'] : '';
  1507. $shop['cartId'] =getArrayItem($shop,'id',0);
  1508. }
  1509. }
  1510. return $data;
  1511. }
  1512. /**
  1513. * 商品信息
  1514. * (原数据基础上+商品名称+商品图片)商品的状态 下架商品计入失效
  1515. * @param $data
  1516. * @return mixed
  1517. * @throws \Exception
  1518. */
  1519. private function formatGoods($data)
  1520. {
  1521. if (empty($data)) return $data;
  1522. $objGoodsBasicRelevantCache = new GoodsBasicRelevant($this->onlineEnterpriseId);
  1523. $objMGoods = new MGoods($this->onlineEnterpriseId, false, $this->onlineUserId);
  1524. $goodsData = $objMGoods->getGoodsNames(array_unique(array_column($data, 'goodsId')));
  1525. if (!empty($goodsData)) {
  1526. $dbResult = $objMGoods->getNameByGoodsIds(array_values(array_unique(array_column($goodsData, 'basicGoodsId'))));
  1527. if ($dbResult->isSuccess()) {
  1528. $basicData = $dbResult->getData();
  1529. foreach ($goodsData as &$goods) {
  1530. $goods['goodsName'] = isset($basicData[$goods['basicGoodsId']]['title']) ? $basicData[$goods['basicGoodsId']]['title'] : '';
  1531. $goods['describe'] = isset($basicData[$goods['basicGoodsId']]['describe']) ? $basicData[$goods['basicGoodsId']]['describe'] : '';
  1532. $goods['categoryId'] = isset($basicData[$goods['basicGoodsId']]['categoryId']) ? $basicData[$goods['basicGoodsId']]['categoryId'] : '';
  1533. $goods['brandId'] = isset($basicData[$goods['basicGoodsId']]['brandId']) ? $basicData[$goods['basicGoodsId']]['brandId'] : '';
  1534. $goods['categoryPath'] = isset($basicData[$goods['basicGoodsId']]['categoryPath']) ? $basicData[$goods['basicGoodsId']]['categoryPath'] : '';
  1535. $goods['specType'] = isset($basicData[$goods['basicGoodsId']]['specType']) ? $basicData[$goods['basicGoodsId']]['specType'] : StatusCode::$specType['single'];
  1536. $goods['storageCode'] = isset($basicData[$goods['basicGoodsId']]['storageCode']) ? $basicData[$goods['basicGoodsId']]['storageCode'] : '';
  1537. $goods['categoryName'] = empty($goods['categoryId']) ? '' : $objGoodsBasicRelevantCache->getNameByCategoryId($goods['categoryId']);
  1538. $goods['brandName'] = empty($goods['brandId']) ? '' : $objGoodsBasicRelevantCache->getNameByBrandId($goods['brandId']);
  1539. $goods['images'] = isset($basicData[$goods['basicGoodsId']]['images']) ? $basicData[$goods['basicGoodsId']]['images'] : [];
  1540. $goods['isEq'] = isset($basicData[$goods['basicGoodsId']]['isEq']) ? $basicData[$goods['basicGoodsId']]['isEq'] : StatusCode::$delete;
  1541. }
  1542. }
  1543. }
  1544. $objMSku = new MSku($this->onlineUserId, $this->onlineEnterpriseId);
  1545. $specNameMapping = $objMSku->getSpecNameBySkuId(array_column($data, 'skuId'));
  1546. if (!$specNameMapping->isSuccess()) {
  1547. $specNameMapping = [];
  1548. } else {
  1549. $specNameMapping = $specNameMapping->getData();
  1550. }
  1551. foreach ($data as $key => &$val) {
  1552. $val['goodsCode'] = createCode(StatusCode::$code['goodsBasic']['prefix'], $val['goodsBasicId'], StatusCode::$code['goodsBasic']['length']);
  1553. $val['brandName'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['brandName'] : '';
  1554. $val['categoryName'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['categoryName'] : '';
  1555. $val['barCode'] = isset($specNameMapping[$val['skuId']]) ? $specNameMapping[$val['skuId']]['barCode'] : '';
  1556. $val['storageCode'] = isset($goodsData[$val['goodsId']]['storageCode']) ? $goodsData[$val['goodsId']]['storageCode'] : '';
  1557. $val['goodsName'] = isset($goodsData[$val['goodsId']]['goodsName']) ? $goodsData[$val['goodsId']]['goodsName'] : '';
  1558. $images = isset($goodsData[$val['goodsId']]['images']) ? $goodsData[$val['goodsId']]['images'] : [];
  1559. $val['goodsImages'] = empty($images) ? '' : $images[0];
  1560. $val['isInvalid'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['enableStatus'] : StatusCode::$delete;//4=>失效商品
  1561. $val['describe'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['describe'] : '';
  1562. $val['categoryId'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['categoryId'] : '';
  1563. $val['brandId'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['brandId'] : '';
  1564. $val['categoryPath'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['categoryPath'] : '';
  1565. $val['isActivity'] = empty($val['activityId']) ? StatusCode::$delete : StatusCode::$standard;//是否是活动商品
  1566. $val['specType'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['specType'] : StatusCode::$specType['single'];
  1567. $val['unitName'] = isset($specNameMapping[$val['skuId']]) ? $specNameMapping[$val['skuId']]['unitName'] : '';
  1568. $val['specGroup'] = isset($specNameMapping[$val['skuId']]) ? $specNameMapping[$val['skuId']]['specGroup'] : [];
  1569. $val['notExpress'] = StatusCode::$standard;
  1570. $val['supplierId'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['supplierId'] : 0;
  1571. $val['supplierName'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['supplierName'] : '';
  1572. $val['isEq'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['isEq'] : StatusCode::$delete;
  1573. $val['isDistribution'] = isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['isDistribution'] : StatusCode::$delete;
  1574. $val['express'] = [
  1575. 'weight' => isset($specNameMapping[$val['skuId']]) ? $specNameMapping[$val['skuId']]['weight'] : 0,
  1576. 'expressType' => isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['expressType'] : 0,
  1577. 'ruleId' => isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['ruleId'] : 0,
  1578. 'expressFee' => isset($goodsData[$val['goodsId']]) ? $goodsData[$val['goodsId']]['expressFee'] : 0,
  1579. ];
  1580. if ($val['isEq'] == StatusCode::$standard){
  1581. $isMaster = isset($specNameMapping[$val['skuId']]) ? $specNameMapping[$val['skuId']]['isMaster'] : StatusCode::$delete;
  1582. //超码商品
  1583. if ($isMaster == StatusCode::$delete){
  1584. $masterSkuResult = $this->objMSku->getMasterById($val['goodsBasicId']);
  1585. if ($masterSkuResult->isSuccess()){
  1586. $masterSku = $masterSkuResult->getData();
  1587. $conversion = isset($specNameMapping[$val['skuId']]) ? $specNameMapping[$val['skuId']]['conversion'] : 0;
  1588. $u = isset($masterSku['unitName']) ? $masterSku['unitName'] : '';
  1589. $u_1 = $val['unitName'];
  1590. $val['extends'] = ['uId' => $masterSku['id'], 'uId_1' => $val['skuId'], 'u' => $u, 'u_1' => $u_1, 'u_1_buy'=> $val['buyNum'],'conversion' => $conversion];
  1591. }
  1592. }
  1593. }
  1594. if (isset($goodsData[$val['goodsId']]) && ($goodsData[$val['goodsId']]['enableStatus'] == StatusCode::$delete || $goodsData[$val['goodsId']]['deleteStatus'] == StatusCode::$delete)) {
  1595. $val['invalidMsg'] = '此商品已下架';
  1596. $this->invalidData[] = $val;//失效商品集合
  1597. unset($data[$key]);
  1598. }
  1599. }
  1600. return $data;
  1601. }
  1602. /**
  1603. * 获取规格信息
  1604. * (原数据+skuData+unitName)
  1605. * @param $data
  1606. * @return mixed
  1607. * @throws \Exception
  1608. */
  1609. private function formatSku($data)
  1610. {
  1611. if (empty($data)) return $data;
  1612. $objMSku = new MSku($this->onlineUserId, $this->onlineEnterpriseId);
  1613. $skuData = $objMSku->getSku(array_unique(array_column($data, 'skuId')));
  1614. if (!empty($skuData)) {
  1615. foreach ($data as &$goods) {
  1616. $goods['skuData'][] = isset($skuData[$goods['skuId']]) ? $skuData[$goods['skuId']] : [];
  1617. $goods['unitName'] = isset($skuData[$goods['skuId']]['unitName']) ? $skuData[$goods['skuId']]['unitName'] : '';
  1618. $goods['conversion'] = isset($skuData[$goods['skuId']]['conversion']) ? $skuData[$goods['skuId']]['conversion'] : '';
  1619. }
  1620. }
  1621. return $data;
  1622. }
  1623. /**
  1624. * 查询库存以及成本(*goodsBasicId *warehouseId) 库存不足的商品计入失效
  1625. * (原数据+costPrice+inventoryNum)
  1626. * @param $data
  1627. * @return mixed
  1628. * @throws \Exception
  1629. */
  1630. private function formatSkuInventory($data): ResultWrapper
  1631. {
  1632. $inventorySelectParams = [];
  1633. $skuIds = [];
  1634. foreach ($data as &$goods) {
  1635. $skuIds[] = $goods['skuId'];
  1636. $goods['inventory'] = 0;
  1637. $inventorySelectParams[$goods['shopId']][] = $goods['skuId'];
  1638. }
  1639. unset($goods);
  1640. $objMInventory = new MInventory($this->onlineEnterpriseId, $this->onlineUserId);
  1641. $inventoryMap = [];
  1642. foreach ($inventorySelectParams as $shopId => $skuIds){
  1643. $inventoryResult = $objMInventory->getInventoryByShopIdAndSkuIds($shopId,$skuIds);
  1644. if (!$inventoryResult->isSuccess()) {
  1645. return ResultWrapper::fail($inventoryResult->getData(),$inventoryResult->getErrorCode());
  1646. }
  1647. $inventoryMap[$shopId] = $inventoryResult->getData();
  1648. }
  1649. var_dump($skuIds) ;
  1650. //获取当前skuIds的主单位,及换算比例
  1651. $skuConversionResult = $this->objMSku->getConversion($skuIds);
  1652. if (!$skuConversionResult->isSuccess()) {
  1653. return ResultWrapper::fail($skuConversionResult->getData(),$skuConversionResult->getErrorCode());
  1654. }
  1655. $skuConversionMapping = $skuConversionResult->getData();
  1656. var_dump($skuConversionMapping);
  1657. var_dump($data);
  1658. exit();
  1659. foreach ($data as $key => &$goods) {
  1660. $thisSku = $skuConversionMapping[$goods['skuId']];
  1661. if (!isset($inventoryMap[$goods['shopId']])){
  1662. return ResultWrapper::fail('未获取到库存信息',ErrorCode::$paramError);
  1663. }
  1664. $row = $inventoryMap[$goods['shopId']];
  1665. $goods['inventoryNum'] = isset($row[$goods['skuId']]['num']) ? $row[$goods['skuId']]['num'] : 0;
  1666. $goods['costPrice'] = 0;
  1667. $goods['conversion'] = $thisSku['conversion'];//换算比例
  1668. //判断是否开启预售
  1669. if ($goods['isDistribution'] == StatusCode::$delete) {
  1670. //没有开启需要检测库存
  1671. if ($goods['inventoryNum'] < $goods['buyNum']) {
  1672. $goods['isInvalid'] = StatusCode::$delete;
  1673. $goods['invalidMsg'] = '商品库存不足';
  1674. $this->invalidData[] = $goods;
  1675. unset($data[$key]);
  1676. }
  1677. }
  1678. }
  1679. return ResultWrapper::success($data);
  1680. }
  1681. /**
  1682. * Doc: (des="计算运费")
  1683. * User: XMing
  1684. * Date: 2020/9/8
  1685. * Time: 10:10 上午
  1686. * @param array $data
  1687. * @return ResultWrapper
  1688. * @throws \Exception
  1689. */
  1690. private function calExpress(array $data)
  1691. {
  1692. if (empty($data)) return ResultWrapper::success($data);
  1693. if (empty($this->onlineUserDefaultDeliveryType)) {
  1694. return ResultWrapper::success($data);
  1695. }
  1696. //判断选择是否是快递配送方式
  1697. if ($this->onlineUserDefaultDeliveryType != StatusCode::$deliveryType['goodsDelivery']) {
  1698. return ResultWrapper::success($data);
  1699. }
  1700. $map = [];
  1701. foreach ($data as $val) {
  1702. $map[$val['shopId']][] = $val;
  1703. }
  1704. unset($data);
  1705. //收货地址为空,按全国计算
  1706. $data = [];
  1707. foreach ($map as $shopId => $item) {
  1708. $buildDataResult = self::buildExpressData($item, $shopId, $this->onlineUserAddressCode);
  1709. if (!$buildDataResult->isSuccess()) {
  1710. return ResultWrapper::fail($buildDataResult->getData(), $buildDataResult->getErrorCode());
  1711. }
  1712. $data = array_merge($data, $buildDataResult->getData());
  1713. }
  1714. return ResultWrapper::success($data);
  1715. }
  1716. /**
  1717. * Doc: (des="运费计算规则")
  1718. * User: XMing
  1719. * Date: 2020/9/8
  1720. * Time: 11:36 上午
  1721. * @example https://www.xiaokeduo.com/help/11584.html
  1722. * 1.按商品累加运费
  1723. * 规则:分别计算出来商品使用模板的运费,和统一运费的最大值,再进行累加。
  1724. * 1)不同或相同的商品,设置同一运费模板:按该模板设置的规则计算
  1725. * 更新:不足续件数目的时候,仍然按照续件数目进行计算。
  1726. * 例如商品A,B都是用模板M(首件10块,续2件5块),如果购买商品A和B,各一件,则一共购买两件,运费=10+5=15元。
  1727. * @param array $data
  1728. * @param string $code
  1729. * @param int $shopId
  1730. * @throws \Exception
  1731. * @return ResultWrapper
  1732. */
  1733. private function buildExpressData(array $data, int $shopId, $code = null)
  1734. {
  1735. //提取设置了运费模版的模版规则id
  1736. $allRuleIds = [];
  1737. foreach ($data as $item) {
  1738. if ($item['express']['expressType'] == StatusCode::$expressType['rule']) {
  1739. $allRuleIds[] = $item['express']['ruleId'];
  1740. continue;
  1741. }
  1742. }
  1743. $allRuleIds = array_values(array_unique($allRuleIds));
  1744. $roleMap = [];
  1745. $objDDeliveryRule = new DDeliveryRule();
  1746. $objDDeliveryRule->setTable('qianniao_delivery_rule_' . $this->onlineEnterpriseId);
  1747. if (!empty($allRuleIds)) {
  1748. $ruleLists = $objDDeliveryRule->select(['id' => $allRuleIds]);
  1749. if ($ruleLists === false) {
  1750. return ResultWrapper::fail($objDDeliveryRule->error(), ErrorCode::$dberror);
  1751. }
  1752. foreach ($ruleLists as $item) {
  1753. $roleMap[$item['id']] = empty($item['setData']) ? [] : json_decode($item['setData'], true);
  1754. }
  1755. }
  1756. // 查询基础设置-商品设置运费规则
  1757. $objMBasicSetup = new MBasicSetup($this->onlineEnterpriseId);
  1758. $BasicSetupResult = $objMBasicSetup->getBasicField('calculateExpressType');
  1759. if (!$BasicSetupResult->isSuccess()) {
  1760. return ResultWrapper::fail($BasicSetupResult->getData(), $BasicSetupResult->getErrorCode());
  1761. }
  1762. $BasicSetup = $BasicSetupResult->getData();
  1763. $calculateExpressType = isset($BasicSetup['calculateExpressType']) ? $BasicSetup['calculateExpressType'] : StatusCode::$standard;
  1764. switch ($calculateExpressType) {
  1765. case StatusCode::$delete: // 累加运费规则
  1766. $calResult = self::ruleAdd($data, $roleMap, $code, $shopId);
  1767. break;
  1768. case StatusCode::$standard: // 组合运费规则
  1769. $calResult = self::ruleGroup($data, $roleMap, $code, $shopId);
  1770. break;
  1771. default: // 默认使用组合运费规则
  1772. $calResult = self::ruleGroup($data, $roleMap, $code, $shopId);
  1773. break;
  1774. }
  1775. if (!$calResult->isSuccess()) {
  1776. return ResultWrapper::fail($calResult->getData(), $calResult->getErrorCode());
  1777. }
  1778. $data = $calResult->getData();
  1779. return ResultWrapper::success($data);
  1780. }
  1781. /**
  1782. * Doc: (des="(规则一)按商品累加运费")
  1783. * User: XMing
  1784. * Date: 2020/9/25
  1785. * Time: 9:14 上午
  1786. * @param array $data
  1787. * @param array $roleMap
  1788. * @param string $code
  1789. * @return ResultWrapper
  1790. * @example https://www.xiaokeduo.com/help/11584.html
  1791. * 1)不同或相同的商品,设置同一运费模板:按该模板设置的规则计算
  1792. * 更新:不足续件数目的时候,仍然按照续件数目进行计算。
  1793. * 例如商品A,B都是用模板M(首件10块,续2件5块),如果购买商品A和B,各一件,则一共购买两件,运费=10+5=15元。
  1794. * 2)多种商品,分别设置不同金额的统一运费:以最高运费金额收取。
  1795. * 更新:例如 商品A,B,C的统一运费分别为1元,2元和3元,一期购买这三个商品,则运费为3元。
  1796. * 3)不同的商品,设置不同的运费模板:分别计算每个运费模板规则应收运费,再累加计算合计运费;
  1797. * 例如: 例如商品A使用用模板M(首件,10块,续1件,5块),商品B使用模板N(首件,12块,续3件,5块),如果购买商品A和B,各2件,则运费=模板M的运费+模板N的运费=(10+5)+(12+5)=32元。
  1798. * 4)统一运费商品,和运费模板商品一同结算:单独计算统一运费商品应收运费,再累加运费模板应收运费;
  1799. * 例如:商品A和B使用统一运费,分别为2元和10元,商品C使用模板M(首1件,10块,续2件,5块),商品D使用模板N(首件,12块,续3件,5块),购买A,B,C和D各两件。
  1800. * 此时统一运费(商品A和B)=10元;运费模板运费(商品C和D)=模板M的运费+模板N的运费=(10+5)+(12+5)=32元。
  1801. * 则总运费=统一运费+运费模板运费=(10)+(32)=42元。
  1802. */
  1803. private function ruleAdd($data, $roleMap, $code, $shopId)
  1804. {
  1805. //初始化所有运费
  1806. $freeExpressMoney = 0;//免邮费
  1807. $ruleExpressMoney = 0;//模版规则运费
  1808. $unifyExpressMoney = 0;//统一运费
  1809. $ruleIdRelGoodsMap = [];//模版=>[商品数量,商品重量]
  1810. foreach ($data as &$item) {
  1811. //判断运费设置
  1812. switch ($item['express']['expressType']) {
  1813. case StatusCode::$expressType['free']:
  1814. //免邮费
  1815. $freeExpressMoney = bcadd($freeExpressMoney, 0, 2);
  1816. break;
  1817. case StatusCode::$expressType['rule']:
  1818. $ruleId = $item['express']['ruleId'];
  1819. //运费模版
  1820. if (!isset($roleMap[$ruleId])) {
  1821. //运费模版规则不存在,不计算运费
  1822. continue 2;
  1823. }
  1824. if (!empty($code)) {
  1825. //地区
  1826. //先判断商品是否在配送区域
  1827. $notDispatchAreas = $roleMap[$ruleId]['not_dispatch_areas'];
  1828. $isExist = self::searchInArray($code, $notDispatchAreas);
  1829. if ($isExist === true) {
  1830. $item['notExpress'] = StatusCode::$delete;
  1831. continue 2;
  1832. }
  1833. }
  1834. $ruleIdRelGoodsMap[$ruleId]['num'] = (isset($ruleIdRelGoodsMap[$ruleId]['num']) ? $ruleIdRelGoodsMap[$ruleId]['num'] : 0) + $item['buyNum'];
  1835. $ruleIdRelGoodsMap[$ruleId]['weight'] = (isset($ruleIdRelGoodsMap[$ruleId]['weight']) ? $ruleIdRelGoodsMap[$ruleId]['weight'] : 0) + bcmul($item['express']['weight'], $item['buyNum'], 2);
  1836. break;
  1837. case StatusCode::$expressType['unify']:
  1838. //统一运费,求其中最大值
  1839. if ($item['express']['expressFee'] > $unifyExpressMoney) {
  1840. //如果当前值大于目前的最大值,则使用当前的值覆盖
  1841. $unifyExpressMoney = $item['express']['expressFee'];
  1842. }
  1843. break;
  1844. default:
  1845. //没有设置默认不计算运费
  1846. break;
  1847. }
  1848. }
  1849. if (!empty($ruleIdRelGoodsMap)) {
  1850. //计算模版运费
  1851. foreach ($ruleIdRelGoodsMap as $ruleId => $value) {
  1852. $type = $roleMap[$ruleId]['type'];//4:按件 5:按重量
  1853. if (empty($code)) {
  1854. //没有地区code,使用全国统一运费的规则进行计算
  1855. $uniform = $roleMap[$ruleId]['uniform'];
  1856. $expressMoney = self::calculateExpress($type, $uniform, $value);
  1857. $ruleExpressMoney = bcadd($ruleExpressMoney, $expressMoney, 2);
  1858. } else {
  1859. //匹配模版
  1860. $dispatchAreas = $roleMap[$ruleId]['areas'];
  1861. $searchRow = self::searchRowByCode($code, $dispatchAreas);
  1862. if ($searchRow === false) {
  1863. //当前地区没有匹配到,使用全国统一计算
  1864. $uniform = $roleMap[$ruleId]['uniform'];
  1865. $expressMoney = self::calculateExpress($type, $uniform, $value);
  1866. $ruleExpressMoney = bcadd($ruleExpressMoney, $expressMoney, 2);
  1867. } else {
  1868. //匹配到了规则
  1869. $rule = [
  1870. 'first_num' => $searchRow['first_num'],
  1871. 'first_price' => $searchRow['first_price'],
  1872. 'second_num' => $searchRow['second_num'],
  1873. 'second_price' => $searchRow['second_price']
  1874. ];
  1875. $expressMoney = self::calculateExpress($type, $rule, $value);
  1876. $ruleExpressMoney = bcadd($ruleExpressMoney, $expressMoney, 2);
  1877. }
  1878. }
  1879. }
  1880. }
  1881. $this->expressShopMoney[$shopId] = bcadd($ruleExpressMoney, $unifyExpressMoney, 2);
  1882. $this->expressMoney = bcadd($this->expressMoney, $this->expressShopMoney[$shopId], 2);
  1883. return ResultWrapper::success($data);
  1884. }
  1885. /**
  1886. * Doc: (des=" (规则二) 组合运费")
  1887. * User: XMing
  1888. * Date: 2020/9/25
  1889. * Time: 12:03 下午
  1890. * @param array $data
  1891. * @param array $roleMap
  1892. * @param string $code
  1893. * @return ResultWrapper
  1894. * @example https://www.xiaokeduo.com/help/11584.html
  1895. * 规则:先将使用统一运费和运费模板的商品分开计算,再取二者较大的值,作为最终运费。
  1896. * 1)计算使用运费模板的运费价格x:取所有商品中,首件金额最大的运费模板,计算使用该模板的所有商品运费;
  1897. * 使用其他模板的所有商品都按照该商品所试用的续件金额来计算;
  1898. * 最后再求和。
  1899. * 2)计算使用统一运费的商品运费y:取最大的统一运费。
  1900. * 3)比较x和y,运费=x和y的较大值。
  1901. * 例如:
  1902. * 1)不同或相同的商品,设置同一运费模板:同按商品累加运费的计算方式,按该模板设置的规则计算。
  1903. * 例如商品A,B都是用模板M(首件10块,续2件5块),如果购买商品A和B,各一件,则一共购买两件,运费=10+5=15元。
  1904. * 2)多种商品,分别设置不同金额的统一运费:同按商品累加运费的计算方式,以最高运费金额收取。
  1905. * 例如 商品A,B,C的统一运费分别为1元,2元和3元,一起购买这三个商品,则运费为3元。
  1906. * 3)不同的商品,设置不同的运费模板:不同于按照商品累加的计算方式。
  1907. * 例如: 例如商品A(1件)使用用模板M(首1件,10块,续1件,5块),商品B(1件)使用模板N(首2件,12块,续1件,5块),如果购买商品A和B,各2件,则运费=模板N的运费(首费为12,大于模板M的10)+模板M的运费(按照续费计算)=(12)+(5*2)=22元。
  1908. * 如果按商品累加计费,则运费=模板M的运费+模板N的运费=(10+5)+(12)=27元。
  1909. * 4)统一运费商品,和运费模板商品一同结算:单独计算统一运费和运费模板,再取较大的数值作为最终运费;
  1910. * 例如:商品A和B使用统一运费,分别为2元和10元,商品C使用模板M(首1件,10块,续2件,5块),商品D(1件)使用模板N(首2件,12块,续1件,5块),购买A,B,C和D各两件。
  1911. * 此时统一运费(商品A和B)=10元;运费模板运费(商品C和D)=模板N的运费(首费为12,大于模板M的10)+模板M的运费(按照续费计算)=(12)+(5*2)=22元。总运费=统一运费和运费模板运费的较大值=22元。
  1912. * 如果按商品累加计费,运费按照之前的计算=统一运费+运费模板运费=(10)+(27)=37元。
  1913. */
  1914. private function ruleGroup($data, $roleMap, $code, $shopId)
  1915. {
  1916. //初始化所有运费
  1917. $freeExpressMoney = 0;//免邮费
  1918. $ruleExpressMoney = 0;//模版规则运费
  1919. $unifyExpressMoney = 0;//统一运费
  1920. $ruleIdRelGoodsMap = [];//模版=>[商品数量,商品重量]
  1921. foreach ($data as &$item) {
  1922. //判断运费设置
  1923. switch ($item['express']['expressType']) {
  1924. case StatusCode::$expressType['free']:
  1925. //免邮费
  1926. $freeExpressMoney = bcadd($freeExpressMoney, 0, 2);
  1927. break;
  1928. case StatusCode::$expressType['rule']:
  1929. $ruleId = $item['express']['ruleId'];
  1930. //运费模版
  1931. if (!isset($roleMap[$ruleId])) {
  1932. //运费模版规则不存在,不计算运费
  1933. continue 2;
  1934. }
  1935. if (!empty($code)) {
  1936. //地区
  1937. //先判断商品是否在配送区域
  1938. $notDispatchAreas = $roleMap[$ruleId]['not_dispatch_areas'];
  1939. $isExist = self::searchInArray($code, $notDispatchAreas);
  1940. if ($isExist === true) {
  1941. $item['notExpress'] = StatusCode::$delete;
  1942. continue 2;
  1943. }
  1944. }
  1945. $ruleIdRelGoodsMap[$ruleId]['num'] = (isset($ruleIdRelGoodsMap[$ruleId]['num']) ? $ruleIdRelGoodsMap[$ruleId]['num'] : 0) + $item['buyNum'];
  1946. $ruleIdRelGoodsMap[$ruleId]['weight'] = (isset($ruleIdRelGoodsMap[$ruleId]['weight']) ? $ruleIdRelGoodsMap[$ruleId]['weight'] : 0) + bcmul($item['express']['weight'], $item['buyNum'], 2);
  1947. break;
  1948. case StatusCode::$expressType['unify']:
  1949. //统一运费,求其中最大值
  1950. if ($item['express']['expressFee'] > $unifyExpressMoney) {
  1951. //如果当前值大于目前的最大值,则使用当前的值覆盖
  1952. $unifyExpressMoney = $item['express']['expressFee'];
  1953. }
  1954. break;
  1955. default:
  1956. //没有设置默认不计算运费
  1957. break;
  1958. }
  1959. }
  1960. //选获取模版规则中首件的最高价格,然后减去首件的都按续件处理
  1961. //唯一的区别,多个运费模版首费只算一次
  1962. if (!empty($ruleIdRelGoodsMap)) {
  1963. //计算模版运费
  1964. foreach ($ruleIdRelGoodsMap as $ruleId => &$value) {
  1965. $value['type'] = $roleMap[$ruleId]['type'];//4:按件 5:按重量
  1966. if (empty($code)) {
  1967. //没有地区code,使用全国统一运费的规则进行计算
  1968. $uniform = $roleMap[$ruleId]['uniform'];
  1969. $value['rule'] = $uniform;
  1970. } else {
  1971. //匹配模版
  1972. $dispatchAreas = $roleMap[$ruleId]['areas'];
  1973. $searchRow = self::searchRowByCode($code, $dispatchAreas);
  1974. if ($searchRow === false) {
  1975. //当前地区没有匹配到,使用全国统一计算
  1976. $uniform = $roleMap[$ruleId]['uniform'];
  1977. $value['rule'] = $uniform;
  1978. } else {
  1979. //匹配到了规则
  1980. $rule = [
  1981. 'first_num' => $searchRow['first_num'],
  1982. 'first_price' => $searchRow['first_price'],
  1983. 'second_num' => $searchRow['second_num'],
  1984. 'second_price' => $searchRow['second_price']
  1985. ];
  1986. $value['rule'] = $rule;
  1987. }
  1988. }
  1989. }
  1990. //使用组合运费计算规则,计算运费
  1991. $ruleExpressMoney = self::calculateExpressGroup($ruleIdRelGoodsMap);
  1992. }
  1993. $this->expressShopMoney[$shopId] = bcadd($ruleExpressMoney, $unifyExpressMoney, 2);
  1994. $this->expressMoney = bcadd($this->expressMoney, $this->expressShopMoney[$shopId], 2);
  1995. return ResultWrapper::success($data);
  1996. }
  1997. /**
  1998. * Doc: (des="以组合运费方式计算快递费用")
  1999. * User: XMing
  2000. * Date: 2020/9/25
  2001. * Time: 2:44 下午
  2002. * @param array $data
  2003. * @return float
  2004. */
  2005. private static function calculateExpressGroup($data)
  2006. {
  2007. $ruleExpressMoney = 0;
  2008. $max_first_price = 0;
  2009. $max_rule_id = 0;
  2010. foreach ($data as $ruleId => $item) {
  2011. //获取首费用最高的
  2012. if ($item['rule']['first_price'] > $max_first_price) {
  2013. $max_first_price = $item['rule']['first_price'];
  2014. $max_rule_id = $ruleId;
  2015. }
  2016. }
  2017. unset($item);
  2018. foreach ($data as $ruleId => &$item) {
  2019. if ($ruleId != $max_rule_id) {
  2020. continue;
  2021. }
  2022. //减去首重的件/重量
  2023. switch ($item['type']) {
  2024. case StatusCode::$delete:
  2025. //剩余续件的
  2026. $num = bcsub($item['num'], $item['rule']['first_num'], 2);
  2027. $item['num'] = ($num < 0) ? 0 : $num;
  2028. break;
  2029. case StatusCode::$standard:
  2030. //剩余续重的
  2031. $weight = bcsub($item['weight'], $item['rule']['first_num'], 2);
  2032. $item['weight'] = ($weight < 0) ? 0 : $weight;
  2033. break;
  2034. }
  2035. }
  2036. unset($item);
  2037. //剩下的都是续
  2038. foreach ($data as $ruleId => $item) {
  2039. switch ($item['type']) {
  2040. case StatusCode::$delete:
  2041. $mod = ($item['rule']['second_num'] == 0) ? 0 : ceil($item['num'] / $item['rule']['second_num']);
  2042. $expressMoney = bcmul($mod, $item['rule']['second_price'], 2);
  2043. $ruleExpressMoney = bcadd($ruleExpressMoney, $expressMoney, 2);
  2044. break;
  2045. case StatusCode::$standard:
  2046. $mod = ($item['rule']['second_num'] == 0) ? 0 : ceil($item['weight'] / $item['rule']['second_num']);
  2047. $expressMoney = bcmul($mod, $item['rule']['second_price'], 2);
  2048. $ruleExpressMoney = bcadd($ruleExpressMoney, $expressMoney, 2);
  2049. break;
  2050. }
  2051. unset($expressMoney);
  2052. unset($mod);
  2053. }
  2054. //最高首费用 + 各自模版续运费
  2055. return bcadd($max_first_price, $ruleExpressMoney, 2);
  2056. }
  2057. /**
  2058. * Doc: (des="根据模版规则,去计算运费")
  2059. * User: XMing
  2060. * Date: 2020/9/25
  2061. * Time: 9:42 上午
  2062. * @param int $type
  2063. * @param array $rule
  2064. * @param array $item
  2065. * @return float
  2066. */
  2067. private static function calculateExpress($type, $rule, $item)
  2068. {
  2069. switch ($type) {
  2070. case StatusCode::$delete:
  2071. //按件
  2072. //1.按件
  2073. if ($item['num'] <= $rule['first_num']) return $rule['first_price'];
  2074. $mod = ($rule['second_num'] == 0) ? 0 : ceil(($item['num'] - $rule['first_num']) / $rule['second_num']);
  2075. return bcadd($rule['first_price'], bcmul($mod, $rule['second_price'], 2), 2);
  2076. break;
  2077. case StatusCode::$standard:
  2078. //按重量
  2079. $thisWeight = bcmul($item['weight'], $item['num'], 2);
  2080. if ($thisWeight <= $rule['first_num']) return $rule['first_price'];
  2081. $mod = ($rule['second_num'] == 0) ? 0 : ceil(($thisWeight - $rule['first_num']) / $rule['second_num']);
  2082. return bcadd($rule['first_price'], bcmul($mod, $rule['second_price'], 2), 2);
  2083. break;
  2084. default:
  2085. break;
  2086. }
  2087. }
  2088. /**
  2089. * Doc: (des="")
  2090. * User: XMing
  2091. * Date: 2020/9/8
  2092. * Time: 3:29 下午
  2093. * @param array $data
  2094. * @param string $code
  2095. * @return array|boolean
  2096. */
  2097. public function searchRowByCode($code, $data)
  2098. {
  2099. $arr = explode('-', $code);
  2100. $provinceCode = isset($arr[0]) ? $arr[0] : 0;
  2101. $cityCode = isset($arr[1]) ? $arr[1] : 0;
  2102. $areaCode = isset($arr[2]) ? $arr[2] : 0;
  2103. foreach ($data as $item) {
  2104. $allProvinces = isset($item['provinces']) ? $item['provinces'] : [];
  2105. $allCitys = isset($item['citys']) ? $item['citys'] : [];
  2106. $allAreas = isset($item['areas']) ? $item['areas'] : [];
  2107. //查看当前省是否存在,如果省不存在市肯定不存在
  2108. if (!in_array($provinceCode, $allProvinces)) {
  2109. continue;
  2110. }
  2111. //查看市是否存在,如果市不存在区肯定不存在
  2112. if (!in_array($cityCode, $allCitys)) {
  2113. continue;
  2114. }
  2115. if (!in_array($areaCode, $allAreas)) {
  2116. continue;
  2117. }
  2118. //存在返回此条数据
  2119. return $item;
  2120. }
  2121. return false;
  2122. }
  2123. /**
  2124. * Doc: (des="查询某只是否在数组中")
  2125. * User: XMing
  2126. * Date: 2020/9/8
  2127. * Time: 3:06 下午
  2128. * @param array $data
  2129. * @param string $code
  2130. * @return boolean 不存在返回false 存在返回true
  2131. */
  2132. public function searchInArray($code, $data)
  2133. {
  2134. $arr = explode('-', $code);
  2135. $provinceCode = isset($arr[0]) ? $arr[0] : 0;
  2136. $cityCode = isset($arr[1]) ? $arr[1] : 0;
  2137. $areaCode = isset($arr[2]) ? $arr[2] : 0;
  2138. $allProvinces = isset($data['provinces']) ? $data['provinces'] : [];
  2139. $allCitys = isset($data['citys']) ? $data['citys'] : [];
  2140. $allAreas = isset($data['areas']) ? $data['areas'] : [];
  2141. //查看当前省是否存在,如果省不存在市肯定不存在
  2142. if (!in_array($provinceCode, $allProvinces)) {
  2143. return false;
  2144. }
  2145. //查看市是否存在,如果市不存在区肯定不存在
  2146. if (!in_array($cityCode, $allCitys)) {
  2147. return false;
  2148. }
  2149. if (!in_array($areaCode, $allAreas)) {
  2150. return false;
  2151. }
  2152. return true;
  2153. }
  2154. /**
  2155. * 获取商品价格,及计算价格 ,(有促销活动的用促销价)
  2156. * (原数据+price+originPrice+preferential+totalMoney)
  2157. * @param $data
  2158. * @return ResultWrapper
  2159. * @throws \Exception
  2160. */
  2161. private function formatPrice($data)
  2162. {
  2163. if (empty($data)) return ResultWrapper::success($data);
  2164. $objMPrice = new MPrice($this->onlineUserId, $this->onlineEnterpriseId);
  2165. $objMActivity = new MActivity($this->onlineUserId, $this->onlineEnterpriseId);
  2166. $objMCustomer = new MCustomer($this->onlineEnterpriseId, $this->onlineUserId);
  2167. $selectParam['material'] = [];
  2168. $allGoodsId = [];
  2169. foreach ($data as &$goods) {
  2170. $allGoodsId[] = $goods['goodsId'];
  2171. $selectParam['material'][$goods['shopId']][] = $goods['goodsId'];
  2172. $goods['originPrice'] = 0;//原价
  2173. $goods['price'] = 0;//单价
  2174. $goods['preferential'] = 0;//优惠差价
  2175. $goods['totalMoney'] = 0;//小计(unitPrice*buyNum)
  2176. $goods['activityMoney'] = 0;//互斥活动商品总金额
  2177. $goods['isMutex'] = StatusCode::$delete;//初始化互斥状态
  2178. }
  2179. unset($goods);
  2180. // 查询价格
  2181. $selectParam['customerId'] = $this->customerId;
  2182. $dbResult = $objMPrice->getPrice($selectParam);
  2183. if (!$dbResult->isSuccess()) {
  2184. return ResultWrapper::fail($dbResult->getData(), $dbResult->getErrorCode());
  2185. }
  2186. $priceResult = $dbResult->getData();
  2187. unset($dbResult);
  2188. $customerResult = $objMCustomer->getCustomerInfoByUserCenterId($this->onlineUserId);
  2189. $activityArr = [];
  2190. if ($customerResult->isSuccess()) {
  2191. $customer = $customerResult->getData();
  2192. //获取促销活动价格
  2193. $dbResult = $objMActivity->getActivity([
  2194. 'goodsId' => implode(',', $allGoodsId),
  2195. 'customerType' => isset($customer['type']) ? $customer['type'] : 0,
  2196. ]);//TODO
  2197. $activityResult = $dbResult->getData();
  2198. unset($dbResult);
  2199. foreach ($activityResult as $activity) {
  2200. $activityArr[$activity['goodsId'] . $activity['skuId']] = $activity;
  2201. }
  2202. }
  2203. //商品价格信息
  2204. $goodsArr = [];
  2205. foreach ($priceResult as $shopPriceArr) {
  2206. foreach ($shopPriceArr as $goodsId => $goodsSkuArr) {
  2207. $goodsArr[$goodsId] = $goodsSkuArr;
  2208. }
  2209. }
  2210. // 查询当前用户是否购买过vip会员卡
  2211. $enableMemberPrice = false;
  2212. $objMVipCard = new MVipCard($this->onlineEnterpriseId, $this->onlineUserId, true);
  2213. $dbResult = $objMVipCard->getAllVipCardByCustomerId($this->customerId);
  2214. if (!$dbResult->isSuccess()) {
  2215. return ResultWrapper::fail($dbResult->getData(), ErrorCode::$dberror);
  2216. }
  2217. $vipCard = $dbResult->getData();
  2218. if (!empty($vipCard)){
  2219. foreach ($vipCard as $key => $value){
  2220. if($value['memberSpecialPrice'] == StatusCode::$standard){
  2221. $enableMemberPrice = true;
  2222. }
  2223. }
  2224. }
  2225. foreach ($data as &$goods) {
  2226. //此商品存在促销价
  2227. if (isset($activityArr[$goods['goodsId'] . $goods['skuId']])) {
  2228. $price = $goodsArr[$goods['goodsId']][$goods['skuId']]['salePrice'];//商品正常销售价格
  2229. //从缓存中获取活动详情
  2230. if (isset($goods['activityId']) && !empty($goods['activityId'])) {
  2231. $price = $activityArr[$goods['goodsId'] . $goods['skuId']]['price'];//商品活动价格
  2232. $activityDetail = $this->objActivityLimitCache->getActivity($goods['activityId']);
  2233. if (!empty($activityDetail) && $activityDetail['isMutex'] == StatusCode::$standard) {
  2234. //开启互斥商品的总金额
  2235. $goods['activityMoney'] = bcmul($price, $goods['buyNum'], 2);
  2236. $goods['isMutex'] = StatusCode::$standard;
  2237. }
  2238. }
  2239. } else {
  2240. if (isset($goodsArr[$goods['goodsId']][$goods['skuId']])) {
  2241. $priceInfo = $goodsArr[$goods['goodsId']][$goods['skuId']];//此商品规格的价格信息
  2242. $price = isset($priceInfo['salePrice']) ? $priceInfo['salePrice'] : 0;
  2243. // 会员价 启用会员价不生效阶梯价了
  2244. $memberPrice = getArrayItem($priceInfo, 'memberPrice', 0);
  2245. if($enableMemberPrice && $memberPrice){
  2246. $price = $memberPrice;
  2247. }else{
  2248. // 开启阶梯价
  2249. if ($priceInfo['enabledLadder']) {
  2250. foreach ($priceInfo['ladderPrice'] as $ladder) {
  2251. $ladder['to'] = empty($ladder['to']) ? 9999 : $ladder['to'];
  2252. if ($goods['buyNum'] >= $ladder['from'] && $goods['buyNum'] <= $ladder['to']) {
  2253. $price = isset($ladder['price']) ? $ladder['price'] : 0;
  2254. continue;
  2255. }
  2256. }
  2257. }
  2258. }
  2259. }
  2260. }
  2261. if (!isset($price)) $price = 0;
  2262. $price = floatval($price);
  2263. $goods['price'] = sprintf("%.2f", $price);
  2264. $goods['originPrice'] = sprintf("%.2f", $price);//没有优惠活动暂时原价=销售价
  2265. $goods['preferential'] = 0;//优惠券优惠金额
  2266. $goods['totalMoney'] = bcmul($price, $goods['buyNum'], 2);//商品小计金额
  2267. $goods['vipDiscount'] = 0;//会员卡优惠金额
  2268. $goods['setNum'] = isset( $goodsArr[$goods['goodsId']][$goods['skuId']]['setNum'] ) ? $goodsArr[$goods['goodsId']][$goods['skuId']]['setNum'] : 0;//起定数量
  2269. if (empty($goods['totalMoney'])) {
  2270. return ResultWrapper::fail('计算商品价格时出现异常', ErrorCode::$paramError);
  2271. }
  2272. }
  2273. return ResultWrapper::success($data);
  2274. }
  2275. /**
  2276. * 数据分组
  2277. *
  2278. * @param $data
  2279. * @param null $userCouponId
  2280. * @param null $vipCardId
  2281. * @return ResultWrapper
  2282. */
  2283. private function formatGroup($data, $userCouponId = null, $vipCardId = null)
  2284. {
  2285. static $checkNum = 0;//购物车中选中商品数量
  2286. static $totalMoney = 0;//原总额
  2287. static $payMoney = 0;//实际支付金额
  2288. static $preferential = 0;//优惠券优惠
  2289. static $cartNum = 0;
  2290. static $goodsNum = 0;//购物车商品数量
  2291. static $vipDiscount = 0;//会员卡优惠金额
  2292. static $vipDoubleDiscount = 0;//会员卡折上折优惠金额
  2293. static $activityMoney = 0;//互斥活动商品总金额
  2294. static $expressMoney = 0;//运费
  2295. $goodsData = [];
  2296. foreach ($data as $key => $val) {
  2297. if (!isset($val['expressMoney'])) {
  2298. $val['expressMoney'] = '0.00';
  2299. }
  2300. $shopGoodsData[$val['shopId']][] = $val;
  2301. if (!isset($goodsData[$val['shopId']]['totalMoney'])) {
  2302. $goodsData[$val['shopId']]['totalMoney'] = '0.00';
  2303. }
  2304. if (!isset($goodsData[$val['shopId']]['preferential'])) {
  2305. $goodsData[$val['shopId']]['preferential'] = '0.00';
  2306. }
  2307. if (!isset($goodsData[$val['shopId']]['expressMoney'])) {
  2308. $goodsData[$val['shopId']]['expressMoney'] = '0.00';
  2309. }
  2310. $goodsData[$val['shopId']] = [
  2311. 'shopId' => $val['shopId'],
  2312. 'shopName' => $val['shopName'],
  2313. 'shopLogo' => $val['shopLogo'],
  2314. 'expressMoney' => isset($this->expressShopMoney[$val['shopId']]) ? $this->expressShopMoney[$val['shopId']] : 0,
  2315. 'totalMoney' => bcadd($goodsData[$val['shopId']]['totalMoney'], $val['totalMoney'], 2),//(购物车总金额)
  2316. 'preferential' => bcadd($goodsData[$val['shopId']]['preferential'], bcmul($val['preferential'], $val['buyNum']), 2),//店铺总优惠金额
  2317. 'payMoney' => bcadd((isset($this->expressShopMoney[$val['shopId']]) ? $this->expressShopMoney[$val['shopId']] : 0), bcsub(bcadd($goodsData[$val['shopId']]['totalMoney'], $val['totalMoney']), bcadd($goodsData[$val['shopId']]['preferential'], bcmul($val['preferential'], $val['buyNum'])), 2), 2),//(购物实付金额)
  2318. 'shopGoodsData' => $shopGoodsData[$val['shopId']],//商品数据
  2319. ];
  2320. //统计选中商品总数
  2321. if ($val['selection'] == StatusCode::$standard) {
  2322. $checkNum = bcadd($checkNum, 1);
  2323. $totalMoney = bcadd($totalMoney, $val['totalMoney'], 2);
  2324. $payMoney = bcadd(0, bcadd($payMoney, bcsub($val['totalMoney'], bcmul($val['buyNum'], $val['preferential'])), 2), 2);
  2325. $preferential = bcadd($preferential, bcmul($val['buyNum'], $val['preferential']), 2);
  2326. $activityMoney = bcadd($activityMoney, $val['activityMoney'], 2);//互斥商品总金额
  2327. }
  2328. $cartNum = bcadd($cartNum, 1);
  2329. $goodsNum = bcadd($goodsNum, $val['buyNum']);
  2330. }
  2331. $result = [
  2332. 'totalMoney' => $totalMoney,
  2333. 'payMoney' => $payMoney,
  2334. 'preferential' => $preferential,
  2335. 'vipDiscount' => $vipDiscount,
  2336. 'vipDoubleDiscount' => $vipDoubleDiscount,
  2337. 'activityMoney' => $activityMoney,
  2338. 'expressMoney' => $this->expressMoney,
  2339. 'checkNum' => (int)$checkNum,
  2340. 'cartNum' => (int)$cartNum,
  2341. 'goodsNum' => (int)$goodsNum,
  2342. 'goodsData' => array_values($goodsData),
  2343. 'invalidData' => array_values($this->invalidData)
  2344. ];
  2345. $objMPreferentialProcess = new MPreferentialProcess($this->onlineUserId, $this->onlineEnterpriseId);
  2346. $result = $objMPreferentialProcess->Preferential($result, $vipCardId, $userCouponId);
  2347. if(!$result->isSuccess()){
  2348. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  2349. }
  2350. $result = $result->getData();
  2351. if (
  2352. (!empty($vipCardId) && $vipCardId != 0)
  2353. || (!empty($userCouponId) && $userCouponId != 0)
  2354. ) {
  2355. //优惠券优惠,会员卡优惠分摊到商品
  2356. $result = self::calAvg($result);
  2357. }
  2358. //处理运费
  2359. $result = self::freeExpressPrice($result);
  2360. $result['expressMoney'] != 0 && $result['payMoney'] = bcadd($result['payMoney'], $result['expressMoney'], 2);
  2361. return ResultWrapper::success($result);
  2362. }
  2363. /**
  2364. * Doc: (des="")
  2365. * User: XMing
  2366. * Date: 2020/9/18
  2367. * Time: 2:56 下午
  2368. * @param $result
  2369. * @return mixed
  2370. */
  2371. private function freeExpressPrice($result)
  2372. {
  2373. $expressMoney = $result['expressMoney'];
  2374. $payMoney = $result['payMoney'];
  2375. $goodsData = $result['goodsData'];
  2376. if ($expressMoney != 0 && $this->freeExpressPrice <= $payMoney && $this->freeExpressPrice != 0) {
  2377. foreach ($goodsData as &$datum) {
  2378. $shopExpressMoney = $datum['expressMoney'];
  2379. $datum['expressMoney'] = '0.00';
  2380. $datum['payMoney'] = bcsub($datum['payMoney'], $shopExpressMoney, 2);
  2381. }
  2382. $expressMoney = '0.00';
  2383. }
  2384. $result['goodsData'] = $goodsData;
  2385. $result['expressMoney'] = $expressMoney;
  2386. return $result;
  2387. }
  2388. /**
  2389. *
  2390. * 优惠计算分摊规则
  2391. * A商品 100元
  2392. * B商品 300元
  2393. * 商品总价 400元
  2394. *
  2395. * A商品享受的优惠 = 优惠券金额*(A商品原价/商品总价)
  2396. * B商品享受的优惠 = 优惠券金额*(B商品原价/商品总价)
  2397. *
  2398. * 优惠券(满100减20) -20
  2399. * 共 -220
  2400. * 实际付款 180
  2401. * A总(220*(100/400)=55) 优惠券(20*(100/400)=5)
  2402. * B总(220*(300/400)=165) 优惠券(20*(300/400)=15)
  2403. * @param $data
  2404. * @return mixed
  2405. */
  2406. private function calAvg($data)
  2407. {
  2408. $account = 0; // 优惠劵优惠金额
  2409. foreach ($data['goodsData'] as &$shops) {
  2410. $count = count($shops['shopGoodsData']);
  2411. foreach ($shops['shopGoodsData'] as $key => &$goods) {
  2412. // 是否为互斥商品
  2413. if ($goods['isMutex'] == StatusCode::$delete) {
  2414. //活动商品不分摊
  2415. if ($data['preferential'] != 0) {
  2416. //有优惠券优惠,分摊金额
  2417. if ($key + 1 != $count) {
  2418. $goods['preferential'] = bcmul($data['preferential'], bcdiv($goods['totalMoney'], $data['totalMoney'], 2), 2);
  2419. $account = bcadd($account, $goods['preferential'], 2);
  2420. } else {
  2421. //最后一个商品分摊金额=总优惠-其他均摊
  2422. $goods['preferential'] = bcsub($data['preferential'], $account, 2);
  2423. }
  2424. }
  2425. }
  2426. if ($data['vipDiscount'] != 0) {
  2427. /**
  2428. * A商品 10 B商品 20 合计30元 折扣1折 A打折9 B打折18 总折扣27
  2429. * A商品折扣金额 = 10 - (10 X (30 - 27)) / 30
  2430. */
  2431. // 会员卡总优惠金额 = 会员卡优惠后的总金额 = 订单实际支付金额
  2432. $vipCardEndDiscountMoney = bcsub($data['totalMoney'],$data['vipDiscount'], 2);
  2433. // 当前商品优惠后的金额 = 当前商品价格 * 会员卡优惠后总金额 / 订单实际支付金额
  2434. $goodsVipCardEndDiscountMoney = bcdiv(bcmul($goods['totalMoney'],$vipCardEndDiscountMoney,2), $data['totalMoney'],2);
  2435. // 当前商品优惠金额 = 当前商品价格 - 当前商品优惠后的金额
  2436. $goods['vipDiscount'] = bcsub($goods['totalMoney'], $goodsVipCardEndDiscountMoney, 2);
  2437. }
  2438. }
  2439. }
  2440. return $data;
  2441. }
  2442. /**
  2443. * 后台--清空用户购物车
  2444. */
  2445. public function clearCart()
  2446. {
  2447. $where['userCenterId'] = $this->onlineUserId;
  2448. $dbResult = $this->objDCart->delete($where);
  2449. if ($dbResult === false) {
  2450. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  2451. }
  2452. return ResultWrapper::success($dbResult);
  2453. }
  2454. /**
  2455. * 批量删除购物车商品
  2456. * @param $params
  2457. * @return ResultWrapper
  2458. */
  2459. public function clearCartByGoodsId($params)
  2460. {
  2461. $where['userCenterId'] = $this->onlineUserId;
  2462. $where['goodsId'] = $params['goodsId'];
  2463. $dbResult = $this->objDCart->delete($where);
  2464. if ($dbResult === false) {
  2465. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  2466. }
  2467. return ResultWrapper::success($dbResult);
  2468. }
  2469. /**
  2470. * 确认订单
  2471. *
  2472. * @param int $userCouponId 优惠券id
  2473. * @param string $vipCardId 会员卡id
  2474. * @param string $addressId
  2475. * @param string $deliveryId
  2476. * @return ResultWrapper
  2477. * @throws \Exception
  2478. */
  2479. public function confirmationOfOrder($userCouponId = '', $vipCardId = '', $addressId = '', $deliveryId = '')
  2480. {
  2481. // 获取客户信息
  2482. $objMCustomer = new MCustomer($this->onlineEnterpriseId, $this->onlineUserId);
  2483. $result = $objMCustomer->getCustomerData(['userCenterId'=>$this->onlineUserId],'*', true);
  2484. if(!$result->isSuccess()){
  2485. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  2486. }
  2487. $customerData = $result->getData();
  2488. if (empty($customerData)) {
  2489. return ResultWrapper::fail('未找到客户信息', ErrorCode::$paramError);
  2490. }
  2491. $customerId = $customerData['id'];
  2492. // 获取客户的默认收货地址
  2493. $objMShippingAddress = new MShippingAddress($this->onlineEnterpriseId);
  2494. $selectParams = [
  2495. 'deleteStatus' => StatusCode::$standard,
  2496. 'defaultStatus' => StatusCode::$standard,
  2497. 'customerId' => $customerId
  2498. ];
  2499. $dbResult = $objMShippingAddress->getShippingAddressInfo($selectParams);
  2500. if (!$dbResult->isSuccess()) {
  2501. return ResultWrapper::fail($dbResult->getData(), ErrorCode::$dberror);
  2502. }
  2503. $addRessData = $dbResult->getData();
  2504. unset($dbResult);
  2505. // 获取自提点
  2506. $objMDeliverySetting = new MDeliverySetting($this->onlineUserId, $this->onlineEnterpriseId);
  2507. $selfExpressRuleResult = $objMDeliverySetting->getAllSelfExpressRule();
  2508. if (!$selfExpressRuleResult->isSuccess()) {
  2509. return ResultWrapper::fail($selfExpressRuleResult->getData(), $selfExpressRuleResult->getErrorCode());
  2510. }
  2511. $selfExpressRule = $selfExpressRuleResult->getData();
  2512. //支付方式
  2513. $objMEnterpriseBindPayment = new MEnterpriseBindPayment($this->onlineUserId, $this->onlineEnterpriseId);
  2514. $paymentData = $objMEnterpriseBindPayment->getPaymentType();
  2515. if (!$paymentData->isSuccess()) {
  2516. $paymentData = [];
  2517. } else {
  2518. $paymentData = $paymentData->getData();
  2519. }
  2520. //配送方式
  2521. $objMDeliverySetting = new MDeliverySetting($this->onlineUserId, $this->onlineEnterpriseId);
  2522. $DeliveryData = $objMDeliverySetting->allDelivery();
  2523. if (!$DeliveryData->isSuccess()) {
  2524. $DeliveryData = [];
  2525. } else {
  2526. $DeliveryData = array_values($DeliveryData->getData()['data']);
  2527. }
  2528. //当前用户选择的配送方式
  2529. if (!empty($deliveryId)) {
  2530. $this->onlineUserDefaultDeliveryType = $deliveryId;
  2531. } else {
  2532. if (!empty($DeliveryData)) {
  2533. foreach ($DeliveryData as $value) {
  2534. if ($value['defaultStatus'] == StatusCode::$standard) {
  2535. $this->onlineUserDefaultDeliveryType = $value['deliveryType'];
  2536. break;
  2537. }
  2538. }
  2539. }
  2540. }
  2541. //当前用户的收货地址code 省-市-区
  2542. if (!empty($addressId)) {
  2543. $addressResult = $objMShippingAddress->getShippingAddressInfo(['id' => $addressId]);
  2544. if (!$addressResult->isSuccess()) {
  2545. return ResultWrapper::fail($addressResult->getData(), ErrorCode::$dberror);
  2546. }
  2547. $addressInfo = $addressResult->getData();
  2548. if (!empty($addressInfo)) {
  2549. $this->onlineUserAddressCode = $addressInfo['provinceCode'] . '-' . $addressInfo['cityCode'] . '-' . $addressInfo['districtCode'];
  2550. }
  2551. } else {
  2552. if (!empty($addRessData)) {
  2553. $this->onlineUserAddressCode = $addRessData['provinceCode'] . '-' . $addRessData['cityCode'] . '-' . $addRessData['districtCode'];
  2554. }
  2555. }
  2556. //获取当前企业设置的包邮价格
  2557. $objMBasicSetup = new MBasicSetup($this->onlineEnterpriseId);
  2558. $freeExpressPriceResult = $objMBasicSetup->getBasicField('freeExpressPrice');
  2559. if (!$freeExpressPriceResult->isSuccess()) {
  2560. return ResultWrapper::fail($freeExpressPriceResult->getData(), $freeExpressPriceResult->getData());
  2561. }
  2562. $freeExpressPrice = $freeExpressPriceResult->getData();
  2563. $this->freeExpressPrice = (isset($freeExpressPrice['freeExpressPrice']) && !empty($freeExpressPrice['freeExpressPrice'])) ? $freeExpressPrice['freeExpressPrice'] : 0;
  2564. //获取购物车选中的数据
  2565. $dbResult = $this->objDCart->select(['selection' => StatusCode::$standard, 'userCenterId' => $this->onlineUserId], 'id,goodsId,goodsCode,buyNum,shopId,goodsBasicId,selection,skuId,warehouseId,activityId', 'createTime DESC');
  2566. if ($dbResult === false) {
  2567. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  2568. }
  2569. if (empty($dbResult)) {
  2570. return ResultWrapper::fail('购物车中没有选定的商品', ErrorCode::$paramError);
  2571. }
  2572. //所有结算商品id
  2573. $goodsIds = array_values(array_column($dbResult, 'goodsId'));
  2574. $cartDataResult = self::formatGoodsAndShop($dbResult, $userCouponId, $vipCardId);
  2575. if (!$cartDataResult->isSuccess()) {
  2576. return ResultWrapper::fail($cartDataResult->getData(), $cartDataResult->getErrorCode());
  2577. }
  2578. $cartData = $cartDataResult->getData();
  2579. if (empty($cartData['goodsData'])) {
  2580. return ResultWrapper::fail('购物车中商品已失效,不能进行结算', ErrorCode::$paramError);
  2581. }
  2582. if (empty($addRessData)) {
  2583. $cartData['address'] = (object)[];
  2584. } else {
  2585. $objMSysAreaChina = new MSysAreaChina();
  2586. $areaName = $objMSysAreaChina->getNameByCode([
  2587. $addRessData['provinceCode'],
  2588. $addRessData['cityCode'],
  2589. $addRessData['districtCode']
  2590. ]);
  2591. $cartData['address'] = [
  2592. 'name' => isset($addRessData['name']) ? $addRessData['name'] : '',
  2593. 'mobile' => isset($addRessData['mobile']) ? $addRessData['mobile'] : '',
  2594. 'address' => isset($addRessData['address']) ? $addRessData['address'] : '',
  2595. 'id' => isset($addRessData['id']) ? $addRessData['id'] : '',
  2596. 'provinceName' => isset($areaName[$addRessData['provinceCode']]) ? $areaName[$addRessData['provinceCode']] : '',
  2597. 'cityName' => isset($areaName[$addRessData['cityCode']]) ? $areaName[$addRessData['cityCode']] : '',
  2598. 'districtName' => isset($areaName[$addRessData['districtCode']]) ? $areaName[$addRessData['districtCode']] : '',
  2599. ];
  2600. }
  2601. //订单使用的优惠券
  2602. $userCouponInfo = [];
  2603. if (!empty($userCouponId)) {
  2604. $objMUserCoupon = new MUserCoupon($this->onlineUserId, $this->onlineEnterpriseId);
  2605. $dbResult = $objMUserCoupon->getUserCoupon(['id' => $userCouponId]);
  2606. if ($dbResult->isSuccess()) $userCouponInfo = $dbResult->getData();
  2607. $userCouponInfo = isset($userCouponInfo[0]) ? $userCouponInfo[0] : [];
  2608. }
  2609. $cartData['useCoupon'] = $userCouponInfo;
  2610. // 订单金额小于银行打款设置的启用金额,不支持银行打款
  2611. if(!empty($paymentData)){
  2612. foreach ($paymentData as $key => $value){
  2613. if($value['id'] == StatusCode::$payType['bankLoans'] && $cartData['payMoney'] < $value['limit']){
  2614. unset($paymentData[$key]);
  2615. }
  2616. }
  2617. }
  2618. $cartData['payment'] = array_values($paymentData);
  2619. $cartData['delivery'] = $DeliveryData;
  2620. $cartData['selfExpressRule'] = $selfExpressRule;
  2621. //优惠券数据
  2622. $cartData = self::findCoupon($cartData);
  2623. //获取订单会员卡
  2624. $vipCard = self::getVipCard($goodsIds);
  2625. if ($vipCard->isSuccess()) {
  2626. $vipCard = $vipCard->getData();
  2627. } else {
  2628. $vipCard = [];
  2629. }
  2630. $cartData['vipCard'][] = $vipCard;
  2631. // 客户余额
  2632. $cartData['balance'] = $customerData['money'];
  2633. // 会员余额
  2634. $cartData['memberBalance'] = $customerData['memberBalance'];
  2635. return ResultWrapper::success($cartData);
  2636. }
  2637. /**
  2638. * @param $goodsIds
  2639. * @return array|ResultWrapper
  2640. */
  2641. public function getVipCard($goodsIds)
  2642. {
  2643. $objMVipCard = new MVipCard($this->onlineEnterpriseId, $this->onlineUserId, true);
  2644. $pageParams = pageToOffset(1, 10);
  2645. $selectParams['limit'] = $pageParams['limit'];
  2646. $selectParams['offset'] = $pageParams['offset'];
  2647. $dbResult = $objMVipCard->getMyVipCards($selectParams);
  2648. if (!$dbResult->isSuccess()) {
  2649. return ResultWrapper::fail($dbResult->getData(), ErrorCode::$dberror);
  2650. }
  2651. $vipCard = $dbResult->getData()['data'];
  2652. if (empty($vipCard)) return ResultWrapper::success([]);
  2653. if (!empty($goodsIds)) {
  2654. foreach ($vipCard as $key => $card) {
  2655. if ($card['mode'] == StatusCode::$delete) {
  2656. $cardGoodsArr = explode(',', $card['goodsIds']);
  2657. foreach ($goodsIds as $goodsId) {
  2658. if (!in_array($goodsId, $cardGoodsArr)) {
  2659. unset($vipCard[$key]);
  2660. }
  2661. }
  2662. }
  2663. }
  2664. }
  2665. $vipCard = empty($vipCard) ? [] : array_shift($vipCard);
  2666. return ResultWrapper::success($vipCard);
  2667. }
  2668. /**
  2669. * 判断在使用范围的店铺,指定商品是否满足最小金额
  2670. * 查询可用的优惠券
  2671. * @param $data
  2672. * @return mixed
  2673. * @throws \Exception
  2674. */
  2675. private function findCoupon($data)
  2676. {
  2677. $objMUserCoupon = new MUserCoupon($this->onlineUserId, $this->onlineEnterpriseId);
  2678. $objGoodsBasicRelevant = new GoodsBasicRelevant($this->onlineEnterpriseId);
  2679. $payAmount = $data['payMoney'];
  2680. if ($data['activityMoney'] != 0) {
  2681. $payAmount = bcsub($payAmount, $data['activityMoney'], 2);
  2682. }
  2683. $dbResult = $objMUserCoupon->availableCoupon([
  2684. 'payAmount' => $payAmount
  2685. ]);
  2686. if (!$dbResult->isSuccess()) {
  2687. return ResultWrapper::fail($dbResult->getData(), ErrorCode::$dberror);
  2688. }
  2689. $coupon = $dbResult->getData();
  2690. foreach ($coupon as $key => $val) {
  2691. $useShop = explode(',', $val['useShop']);//此优惠券可以使用的店铺
  2692. $applyRange = $val['applyRange'];//优惠券使用范围
  2693. $cart[$val['id']]['money'] = 0;
  2694. foreach ($data['goodsData'] as $shopGoodsData) {
  2695. foreach ($shopGoodsData['shopGoodsData'] as $goodsDetail) {
  2696. $categoryId = $objGoodsBasicRelevant->getNameByBasicId($goodsDetail['goodsBasicId'], 'categoryId');
  2697. $brandId = $objGoodsBasicRelevant->getNameByBasicId($goodsDetail['goodsBasicId'], 'brandId');
  2698. if (in_array($shopGoodsData['shopId'], $useShop) &&
  2699. ($applyRange == StatusCode::$applyRange['allGoods'] ||
  2700. in_array($categoryId, explode(',', $val['categoryCollect'])) ||
  2701. in_array($goodsDetail['goodsId'], explode(',', $val['goodsCollect'])) ||
  2702. in_array($brandId, explode(',', $val['brandCollect'])))
  2703. ) {
  2704. $cart[$val['id']]['money'] = bcadd($cart[$val['id']]['money'], $goodsDetail['totalMoney']);//规定范围内的商品总金额
  2705. }
  2706. }
  2707. }
  2708. if ($val['minPrice'] != 0 && $cart[$val['id']]['money'] < $val['minPrice']) {
  2709. unset($coupon[$key]);//不符合要求
  2710. }
  2711. }
  2712. $data['coupon'] = array_values($coupon);
  2713. return $data;
  2714. }
  2715. /**
  2716. * 获取商品,规格 购物车数量
  2717. * @param $goodsIds
  2718. * @return array
  2719. */
  2720. public function getCartNumByGoodsId($goodsIds)
  2721. {
  2722. $cartData = $this->objDCart->select(['goodsId' => $goodsIds, 'userCenterId' => $this->onlineUserId], 'buyNum,goodsId,skuId');
  2723. if ($cartData === false) {
  2724. return [];
  2725. }
  2726. $return = [];
  2727. foreach ($cartData as $key => $val) {
  2728. $return[$val['goodsId']][$val['skuId']] = $val['buyNum'];
  2729. }
  2730. return $return;
  2731. }
  2732. /**
  2733. * 商品列表页调整购买数量
  2734. * @param $params
  2735. * @return ResultWrapper
  2736. * @throws \Exception
  2737. */
  2738. public function updateCartNum($params)
  2739. {
  2740. if ($params['buyNum'] == 0) {
  2741. $dbResult = $this->objDCart->delete([
  2742. 'goodsId' => $params['goodsId'],
  2743. 'skuId' => $params['skuId'],
  2744. 'userCenterId' => $this->onlineUserId
  2745. ]);
  2746. } else {
  2747. $goods = self::getCartInventory([
  2748. 'goodsId' => $params['goodsId'],
  2749. 'skuId' => $params['skuId'],
  2750. 'userCenterId' => $this->onlineUserId
  2751. ]);
  2752. if (!$goods->isSuccess()) {
  2753. return ResultWrapper::fail($goods->getData(), ErrorCode::$dberror);
  2754. }
  2755. $goods = $goods->getData();
  2756. $inventoryNum = isset($goods[0]['inventory']) ? $goods[0]['inventory'] : 0;//商品剩余库存
  2757. $setNum = isset($goods[0]['setNum']) ? $goods[0]['setNum'] : 0;//商品规格起订数量
  2758. $title = isset($goods[0]['title']) ? $goods[0]['title'] : '';//商品名称
  2759. if ($inventoryNum < $params['buyNum']) {
  2760. return ResultWrapper::fail('商品库存不足', ErrorCode::$paramError);
  2761. }
  2762. if ($params['buyNum'] < $setNum) {
  2763. return ResultWrapper::fail($title . '最小起订数量' . $setNum, ErrorCode::$paramError);
  2764. }
  2765. $dbResult = $this->objDCart->update(
  2766. [
  2767. 'buyNum' => $params['buyNum']
  2768. ],
  2769. [
  2770. 'goodsId' => $params['goodsId'],
  2771. 'skuId' => $params['skuId'],
  2772. 'userCenterId' => $this->onlineUserId
  2773. ]
  2774. );
  2775. }
  2776. if ($dbResult === false) {
  2777. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  2778. }
  2779. return ResultWrapper::success($dbResult);
  2780. }
  2781. /**
  2782. * 验证购物车内商品规格的库存
  2783. * @param $selectParams
  2784. * @return int|ResultWrapper
  2785. * @throws \Exception
  2786. */
  2787. private function getCartInventory($selectParams)
  2788. {
  2789. $dbResult = $this->objDCart->get($selectParams);
  2790. if ($dbResult === false) {
  2791. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  2792. }
  2793. if (empty($dbResult)) {
  2794. return ResultWrapper::success([]);
  2795. }
  2796. $result = self::checkCart([$dbResult]);
  2797. if (!$result->isSuccess()) {
  2798. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  2799. }
  2800. return ResultWrapper::success($result->getData());
  2801. }
  2802. /**
  2803. * Doc: (des="")
  2804. * User: XMing
  2805. * Date: 2021/1/8
  2806. * Time: 10:30 上午
  2807. * @param int $skuId
  2808. * @param int $userId
  2809. * @return ResultWrapper
  2810. */
  2811. public function getCartBuyNumBySku(int $skuId, int $userId): ResultWrapper
  2812. {
  2813. $sql = 'SELECT SUM(`buyNum`) AS subBuyNum FROM qianniao_cart_' . $this->onlineEnterpriseId . ' WHERE skuId = ' . $skuId . ' AND userCenterId = ' . $userId;
  2814. $sum = $this->objDCart->query($sql);
  2815. if ($sum === false) {
  2816. return ResultWrapper::fail($this->objDCart->error(), ErrorCode::$dberror);
  2817. }
  2818. $subBuyNum = isset($sum[0]['subBuyNum']) ? $sum[0]['subBuyNum'] : 0;
  2819. return ResultWrapper::success($subBuyNum);
  2820. }
  2821. /**
  2822. * Doc: (des="组团立即购买")
  2823. * User: XMing
  2824. * Date: 2021/1/22
  2825. * Time: 4:18 下午
  2826. * @param array $lists
  2827. * @param array $data
  2828. * @return ResultWrapper
  2829. * @throws \Exception
  2830. */
  2831. public function buyNow(array $lists, array $data): ResultWrapper
  2832. {
  2833. switch ($data['sourceType']) {
  2834. case self::$sourceType['comBin']:
  2835. $buyResult = self::buyComBin($lists, $data);
  2836. break;
  2837. }
  2838. if (!$buyResult->isSuccess()) {
  2839. return ResultWrapper::fail($buyResult->getData(), $buyResult->getErrorCode());
  2840. }
  2841. return ResultWrapper::success($buyResult->getData());
  2842. }
  2843. /**
  2844. * 普通商品立即购买
  2845. */
  2846. public function goodsBuyNow(array $goods, array $data)
  2847. {
  2848. //获取当前用户的客户id
  2849. $objMCustomer = new MCustomer($this->onlineEnterpriseId, $this->onlineUserId);
  2850. $result = $objMCustomer->getCustomerData(['userCenterId'=>$this->onlineUserId],'*', true);
  2851. if(!$result->isSuccess()){
  2852. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  2853. }
  2854. $customerData = $result->getData();
  2855. if (empty($customerData)) {
  2856. return ResultWrapper::fail('未找到客户信息', ErrorCode::$paramError);
  2857. }
  2858. $customerId = $customerData['id'];
  2859. //获取客户的默认收货地址
  2860. $objMShippingAddress = new MShippingAddress($this->onlineEnterpriseId);
  2861. $selectParams = [
  2862. 'deleteStatus' => StatusCode::$standard,
  2863. 'defaultStatus' => StatusCode::$standard,
  2864. 'customerId' => $customerId
  2865. ];
  2866. $dbResult = $objMShippingAddress->getShippingAddressInfo($selectParams);
  2867. if (!$dbResult->isSuccess()) {
  2868. return ResultWrapper::fail($dbResult->getData(), ErrorCode::$dberror);
  2869. }
  2870. $addRessData = $dbResult->getData();
  2871. unset($dbResult);
  2872. //获取自提点
  2873. $objMDeliverySetting = new MDeliverySetting($this->onlineUserId, $this->onlineEnterpriseId);
  2874. $selfExpressRuleResult = $objMDeliverySetting->getAllSelfExpressRule();
  2875. if (!$selfExpressRuleResult->isSuccess()) {
  2876. return ResultWrapper::fail($selfExpressRuleResult->getData(), $selfExpressRuleResult->getErrorCode());
  2877. }
  2878. $selfExpressRule = $selfExpressRuleResult->getData();
  2879. //支付方式
  2880. $objMEnterpriseBindPayment = new MEnterpriseBindPayment($this->onlineUserId, $this->onlineEnterpriseId);
  2881. $paymentData = $objMEnterpriseBindPayment->getPaymentType();
  2882. if (!$paymentData->isSuccess()) {
  2883. $paymentData = [];
  2884. } else {
  2885. $paymentData = $paymentData->getData();
  2886. }
  2887. //配送方式
  2888. $objMDeliverySetting = new MDeliverySetting($this->onlineUserId, $this->onlineEnterpriseId);
  2889. $DeliveryData = $objMDeliverySetting->allDelivery();
  2890. if (!$DeliveryData->isSuccess()) {
  2891. $DeliveryData = [];
  2892. } else {
  2893. $DeliveryData = array_values($DeliveryData->getData()['data']);
  2894. }
  2895. //当前用户选择的配送方式
  2896. if (!empty($data['deliveryId'])) {
  2897. $this->onlineUserDefaultDeliveryType = $data['deliveryId'];
  2898. } else {
  2899. if (!empty($DeliveryData)) {
  2900. foreach ($DeliveryData as $value) {
  2901. if ($value['defaultStatus'] == StatusCode::$standard) {
  2902. $this->onlineUserDefaultDeliveryType = $value['deliveryType'];
  2903. break;
  2904. }
  2905. }
  2906. }
  2907. }
  2908. //当前用户的收货地址code 省-市-区
  2909. if (!empty($addressId)) {
  2910. $addressResult = $objMShippingAddress->getShippingAddressInfo(['id' => $data['addressId']]);
  2911. if (!$addressResult->isSuccess()) {
  2912. return ResultWrapper::fail($addressResult->getData(), ErrorCode::$dberror);
  2913. }
  2914. $addressInfo = $addressResult->getData();
  2915. if (!empty($addressInfo)) {
  2916. $this->onlineUserAddressCode = $addressInfo['provinceCode'] . '-' . $addressInfo['cityCode'] . '-' . $addressInfo['districtCode'];
  2917. }
  2918. } else {
  2919. if (!empty($addRessData)) {
  2920. $this->onlineUserAddressCode = $addRessData['provinceCode'] . '-' . $addRessData['cityCode'] . '-' . $addRessData['districtCode'];
  2921. }
  2922. }
  2923. //获取当前企业设置的包邮价格
  2924. $objMBasicSetup = new MBasicSetup($this->onlineEnterpriseId);
  2925. $freeExpressPriceResult = $objMBasicSetup->getBasicField('freeExpressPrice');
  2926. if (!$freeExpressPriceResult->isSuccess()) {
  2927. return ResultWrapper::fail($freeExpressPriceResult->getData(), $freeExpressPriceResult->getData());
  2928. }
  2929. $freeExpressPrice = $freeExpressPriceResult->getData();
  2930. $this->freeExpressPrice = (isset($freeExpressPrice['freeExpressPrice']) && !empty($freeExpressPrice['freeExpressPrice'])) ? $freeExpressPrice['freeExpressPrice'] : 0;
  2931. //所有结算商品id
  2932. $goodsIds = [$goods['goodsId']];
  2933. $cartDataResult = self::formatGoodsAndShop([$goods], $data['couponId'], $data['vipCardId']);
  2934. if (!$cartDataResult->isSuccess()) {
  2935. return ResultWrapper::fail($cartDataResult->getData(), $cartDataResult->getErrorCode());
  2936. }
  2937. $cartData = $cartDataResult->getData();
  2938. if (empty($cartData['goodsData'])) {
  2939. return ResultWrapper::fail('商品已失效,不能进行结算', ErrorCode::$paramError);
  2940. }
  2941. if (empty($addRessData)) {
  2942. $cartData['address'] = (object)[];
  2943. } else {
  2944. $objMSysAreaChina = new MSysAreaChina();
  2945. $areaName = $objMSysAreaChina->getNameByCode([
  2946. $addRessData['provinceCode'],
  2947. $addRessData['cityCode'],
  2948. $addRessData['districtCode']
  2949. ]);
  2950. $cartData['address'] = [
  2951. 'name' => isset($addRessData['name']) ? $addRessData['name'] : '',
  2952. 'mobile' => isset($addRessData['mobile']) ? $addRessData['mobile'] : '',
  2953. 'address' => isset($addRessData['address']) ? $addRessData['address'] : '',
  2954. 'id' => isset($addRessData['id']) ? $addRessData['id'] : '',
  2955. 'provinceName' => isset($areaName[$addRessData['provinceCode']]) ? $areaName[$addRessData['provinceCode']] : '',
  2956. 'cityName' => isset($areaName[$addRessData['cityCode']]) ? $areaName[$addRessData['cityCode']] : '',
  2957. 'districtName' => isset($areaName[$addRessData['districtCode']]) ? $areaName[$addRessData['districtCode']] : '',
  2958. ];
  2959. }
  2960. //订单使用的优惠券
  2961. $userCouponInfo = [];
  2962. if (!empty($data['couponId'])) {
  2963. $objMUserCoupon = new MUserCoupon($this->onlineUserId, $this->onlineEnterpriseId);
  2964. $dbResult = $objMUserCoupon->getUserCoupon(['id' => $data['couponId']]);
  2965. if ($dbResult->isSuccess()) $userCouponInfo = $dbResult->getData();
  2966. $userCouponInfo = isset($userCouponInfo[0]) ? $userCouponInfo[0] : [];
  2967. }
  2968. // 订单金额小于银行打款设置的启用金额,不支持银行打款
  2969. if(!empty($paymentData)){
  2970. foreach ($paymentData as $key => $value){
  2971. if($value['id'] == StatusCode::$payType['bankLoans'] && $cartData['payMoney'] <= $value['limit']){
  2972. unset($paymentData[$key]);
  2973. }
  2974. }
  2975. }
  2976. $cartData['payment'] = array_values($paymentData);
  2977. $cartData['useCoupon'] = $userCouponInfo;
  2978. $cartData['delivery'] = $DeliveryData;
  2979. $cartData['selfExpressRule'] = $selfExpressRule;
  2980. //优惠券数据
  2981. $cartData = self::findCoupon($cartData);
  2982. //获取订单会员卡
  2983. $vipCard = self::getVipCard($goodsIds);
  2984. if ($vipCard->isSuccess()) {
  2985. $vipCard = $vipCard->getData();
  2986. } else {
  2987. $vipCard = [];
  2988. }
  2989. $cartData['vipCard'][] = $vipCard;
  2990. // 客户余额
  2991. $cartData['balance'] = $customerData['money'];
  2992. // 会员余额
  2993. $cartData['memberBalance'] = $customerData['memberBalance'];
  2994. return ResultWrapper::success($cartData);
  2995. }
  2996. /**
  2997. * Doc: (des="购买组合套餐")
  2998. * User: XMing
  2999. * Date: 2021/1/22
  3000. * Time: 4:34 下午
  3001. * @param array $lists
  3002. * @param array $data
  3003. * @return ResultWrapper
  3004. * @throws \Exception
  3005. */
  3006. private function buyComBin(array $lists, array $data): ResultWrapper
  3007. {
  3008. $objMCustomer = new MCustomer($this->onlineEnterpriseId,$this->onlineUserId);
  3009. $customerId = $objMCustomer->getCustomerIdByUserCenterId($this->onlineUserId);
  3010. //验证组合套餐
  3011. if (empty($data['comBinId'])) {
  3012. return ResultWrapper::fail('组合套餐错误', ErrorCode::$paramError);
  3013. }
  3014. $objMComBinPackage = new MComBinPackage($this->onlineEnterpriseId, $this->onlineUserId);
  3015. $detailsResult = $objMComBinPackage->details($data['comBinId']);
  3016. if (!$detailsResult->isSuccess()) {
  3017. return ResultWrapper::fail($detailsResult->getData(), $detailsResult->getErrorCode());
  3018. }
  3019. $details = $detailsResult->getData();
  3020. if (empty($details)) {
  3021. return ResultWrapper::fail('未获取到指定的组合套餐', ErrorCode::$paramError);
  3022. }
  3023. $time = time();
  3024. if ($details['isExpire'] == StatusCode::$standard) {
  3025. if ($time < $details['startTime']) {
  3026. return ResultWrapper::fail('组合套餐活动未开始', ErrorCode::$paramError);
  3027. }
  3028. if ($time > $details['endTime']) {
  3029. return ResultWrapper::fail('组合套餐活动已结束', ErrorCode::$paramError);
  3030. }
  3031. }
  3032. if ($details['isLimit'] == StatusCode::$standard) {
  3033. //获取用户组合套餐购买数量(TODO)
  3034. $user_log_num = 0;
  3035. if ($user_log_num >= $details['limitNum']) {
  3036. return ResultWrapper::fail('你已经购买过此组合套餐了', ErrorCode::$paramError);
  3037. }
  3038. }
  3039. //验证商品在售状态,sku是否参与组合,库存
  3040. $allowSkuArr = [];
  3041. $goodsData = $details['goodsData'];
  3042. $mapping = [];//商品映射
  3043. foreach ($goodsData as $goods) {
  3044. $sku = $goods['joinSku'];
  3045. $allowSkuArr = array_merge($sku, $allowSkuArr);
  3046. $mapping[$goods['id']] = $goods;
  3047. }
  3048. //验证商品完整性
  3049. $goodsIdArr = explode(',', $details['goodsIds']);
  3050. $selectGoodsIdArr = [];
  3051. foreach ($lists as $val) {
  3052. if (!in_array($val['goodsId'], $selectGoodsIdArr)) {
  3053. $selectGoodsIdArr[] = $val['goodsId'];
  3054. }
  3055. $thisRow = getArrayItem($mapping,$val['goodsId'],[]);
  3056. if (isset($thisRow['joinSku']) && !empty($thisRow['joinSku'])){
  3057. if (!in_array($val['skuId'], $allowSkuArr)) {
  3058. return ResultWrapper::fail('商品规格选择异常', ErrorCode::$paramError);
  3059. }
  3060. }
  3061. }
  3062. $diffBool = array_diff($goodsIdArr, $selectGoodsIdArr);
  3063. if (!empty($diffBool)) {
  3064. return ResultWrapper::fail('你还没选择完组合商品', ErrorCode::$paramError);
  3065. }
  3066. unset($diffBool);
  3067. //店铺
  3068. $objMGoods = new MShop($this->onlineEnterpriseId, $this->onlineUserId);
  3069. $shopResult = $objMGoods->getShopByIdd($details['shopId']);
  3070. if (!$shopResult->isSuccess()) {
  3071. return ResultWrapper::fail($shopResult->getData(), $shopResult->getErrorCode());
  3072. }
  3073. $shop = $shopResult->getData();
  3074. $c_goods_Lists = [];
  3075. foreach ($lists as $val) {
  3076. if (!isset($mapping[$val['goodsId']])) {
  3077. return ResultWrapper::fail('商品信息异常', ErrorCode::$paramError);
  3078. }
  3079. $row = getArrayItem($mapping, $val['goodsId'], []);
  3080. $specMultiple = getArrayItem($row, 'specMultiple', []);
  3081. $specMultipleMap = [];//sku映射
  3082. foreach ($specMultiple as $value) {
  3083. $specMultipleMap[$value['id']] = $value;
  3084. }
  3085. if ($row['enableStatus'] == StatusCode::$delete) {
  3086. return ResultWrapper::fail('组合套餐商品下架了', ErrorCode::$paramError);
  3087. }
  3088. $skuRow = getArrayItem($specMultipleMap, $val['skuId'], []);
  3089. if ($this->preSale == StatusCode::$delete) {
  3090. if ($skuRow['inventory'] < 1) {
  3091. return ResultWrapper::fail('组合套餐库存不足', ErrorCode::$paramError);
  3092. }
  3093. }
  3094. $c_goods_Lists[] = [
  3095. 'goodsId' => getArrayItem($row, 'id', 0),
  3096. 'goodsCode' => createCode(StatusCode::$code['goodsBasic']['prefix'], $row['basicGoodsId'], StatusCode::$code['goodsBasic']['length']),
  3097. 'buyNum' => 1,
  3098. 'shopId' => getArrayItem($row, 'shopId', 0),
  3099. 'shopLogo' => getArrayItem($shop, 'logo', ''),
  3100. 'goodsBasicId' => getArrayItem($row, 'basicGoodsId', 0),
  3101. 'selection' => StatusCode::$standard,
  3102. 'skuId' => $val['skuId'],
  3103. 'warehouseId' => getArrayItem($row, 'warehouseId', 0),
  3104. 'activityId' => 0,
  3105. 'shopName' => getArrayItem($row, 'shopName', ''),
  3106. 'brandName' => getArrayItem($row, 'brandName', ''),
  3107. 'categoryName' => getArrayItem($row, 'categoryName', ''),
  3108. 'barCode' => getArrayItem($skuRow, 'barCode', ''),
  3109. 'storageCode' => getArrayItem($row, 'storageCode', ''),
  3110. 'goodsName' => getArrayItem($row, 'title', ''),
  3111. 'goodsImages' => array_shift($row['images']),
  3112. 'isInvalid' => StatusCode::$standard,
  3113. 'describe' => getArrayItem($row, 'describe', ''),
  3114. 'categoryId' => getArrayItem($row, 'categoryId', 0),
  3115. 'brandId' => getArrayItem($row, 'brandId', 0),
  3116. 'categoryPath' => getArrayItem($row, 'categoryPath', ''),
  3117. 'isActivity' => StatusCode::$delete,
  3118. 'specType' => getArrayItem($row, 'specType', 0),
  3119. 'unitName' => getArrayItem($skuRow, 'unitName', ''),
  3120. 'specGroup' => getArrayItem($skuRow, 'specGroup', []),
  3121. 'notExpress' => StatusCode::$standard,
  3122. 'supplierId' => getArrayItem($row, 'supplierId', 0),
  3123. 'express' => [
  3124. "weight" => getArrayItem($skuRow, 'weight', 0),
  3125. "expressType" => getArrayItem($row, 'expressType', 0),
  3126. "ruleId" => getArrayItem($row, 'ruleId', 0),
  3127. "expressFee" => getArrayItem($row, 'expressFee', 0),
  3128. ],
  3129. 'originPrice' => getArrayItem($skuRow, 'marketPrice', 0),
  3130. 'price' => getArrayItem($skuRow, 'salePrice', 0),
  3131. 'preferential' => 0,
  3132. 'totalMoney' => getArrayItem($skuRow, 'salePrice', 0),
  3133. 'activityMoney' => 0,
  3134. 'isMutex' => StatusCode::$delete,
  3135. 'costPrice' => getArrayItem($skuRow, 'costPrice', 0),
  3136. 'inventoryNum' => getArrayItem($skuRow, 'inventory', 0),
  3137. 'conversion' => getArrayItem($skuRow, 'conversion', 0),
  3138. 'expressMoney' => 0,
  3139. ];
  3140. }
  3141. //获取客户的默认收货地址
  3142. $objMShippingAddress = new MShippingAddress($this->onlineEnterpriseId);
  3143. $selectParams = [
  3144. 'deleteStatus' => StatusCode::$standard,
  3145. 'defaultStatus' => StatusCode::$standard,
  3146. 'customerId' => $customerId
  3147. ];
  3148. $dbResult = $objMShippingAddress->getShippingAddressInfo($selectParams);
  3149. if (!$dbResult->isSuccess()) {
  3150. return ResultWrapper::fail($dbResult->getData(), ErrorCode::$dberror);
  3151. }
  3152. $addRessData = $dbResult->getData();
  3153. unset($dbResult);
  3154. //获取自提点
  3155. $objMDeliverySetting = new MDeliverySetting($this->onlineUserId, $this->onlineEnterpriseId);
  3156. $selfExpressRuleResult = $objMDeliverySetting->getAllSelfExpressRule();
  3157. if (!$selfExpressRuleResult->isSuccess()) {
  3158. return ResultWrapper::fail($selfExpressRuleResult->getData(), $selfExpressRuleResult->getErrorCode());
  3159. }
  3160. $selfExpressRule = $selfExpressRuleResult->getData();
  3161. //支付方式
  3162. $objMEnterpriseBindPayment = new MEnterpriseBindPayment($this->onlineUserId, $this->onlineEnterpriseId);
  3163. $paymentData = $objMEnterpriseBindPayment->getPaymentType();
  3164. if (!$paymentData->isSuccess()) {
  3165. return ResultWrapper::fail($paymentData->getData(),$paymentData->getErrorCode());
  3166. }
  3167. $paymentData = $paymentData->getData();
  3168. //配送方式
  3169. $objMDeliverySetting = new MDeliverySetting($this->onlineUserId, $this->onlineEnterpriseId);
  3170. $DeliveryData = $objMDeliverySetting->allDelivery();
  3171. if (!$DeliveryData->isSuccess()) {
  3172. return ResultWrapper::fail($DeliveryData->getData(),$DeliveryData->getErrorCode());
  3173. }
  3174. $DeliveryData = array_values($DeliveryData->getData()['data']);
  3175. $deliveryId = getArrayItem($data,'deliveryId',0);
  3176. //当前用户选择的配送方式
  3177. if (!empty($deliveryId)) {
  3178. $this->onlineUserDefaultDeliveryType = $deliveryId;
  3179. } else {
  3180. if (!empty($DeliveryData)) {
  3181. foreach ($DeliveryData as $value) {
  3182. if ($value['defaultStatus'] == StatusCode::$standard) {
  3183. $this->onlineUserDefaultDeliveryType = $value['deliveryType'];
  3184. break;
  3185. }
  3186. }
  3187. }
  3188. }
  3189. //当前用户的收货地址code 省-市-区
  3190. $addressId = getArrayItem($data,'addressId',0);
  3191. if (!empty($addressId)) {
  3192. $addressResult = $objMShippingAddress->getShippingAddressInfo(['id' => $addressId]);
  3193. if (!$addressResult->isSuccess()) {
  3194. return ResultWrapper::fail($addressResult->getData(), ErrorCode::$dberror);
  3195. }
  3196. $addressInfo = $addressResult->getData();
  3197. if (!empty($addressInfo)) {
  3198. $this->onlineUserAddressCode = $addressInfo['provinceCode'] . '-' . $addressInfo['cityCode'] . '-' . $addressInfo['districtCode'];
  3199. }
  3200. } else {
  3201. if (!empty($addRessData)) {
  3202. $this->onlineUserAddressCode = $addRessData['provinceCode'] . '-' . $addRessData['cityCode'] . '-' . $addRessData['districtCode'];
  3203. }
  3204. }
  3205. if (empty($addRessData)) {
  3206. $address = (object)[];
  3207. } else {
  3208. $objMSysAreaChina = new MSysAreaChina();
  3209. $areaName = $objMSysAreaChina->getNameByCode([
  3210. $addRessData['provinceCode'],
  3211. $addRessData['cityCode'],
  3212. $addRessData['districtCode']
  3213. ]);
  3214. $address = [
  3215. 'name' => isset($addRessData['name']) ? $addRessData['name'] : '',
  3216. 'mobile' => isset($addRessData['mobile']) ? $addRessData['mobile'] : '',
  3217. 'address' => isset($addRessData['address']) ? $addRessData['address'] : '',
  3218. 'id' => isset($addRessData['id']) ? $addRessData['id'] : '',
  3219. 'provinceName' => isset($areaName[$addRessData['provinceCode']]) ? $areaName[$addRessData['provinceCode']] : '',
  3220. 'cityName' => isset($areaName[$addRessData['cityCode']]) ? $areaName[$addRessData['cityCode']] : '',
  3221. 'districtName' => isset($areaName[$addRessData['districtCode']]) ? $areaName[$addRessData['districtCode']] : '',
  3222. ];
  3223. }
  3224. //获取当前企业设置的包邮价格
  3225. $objMBasicSetup = new MBasicSetup($this->onlineEnterpriseId);
  3226. $freeExpressPriceResult = $objMBasicSetup->getBasicField('freeExpressPrice');
  3227. if (!$freeExpressPriceResult->isSuccess()) {
  3228. return ResultWrapper::fail($freeExpressPriceResult->getData(), $freeExpressPriceResult->getData());
  3229. }
  3230. $freeExpressPrice = $freeExpressPriceResult->getData();
  3231. $this->freeExpressPrice = (isset($freeExpressPrice['freeExpressPrice']) && !empty($freeExpressPrice['freeExpressPrice'])) ? $freeExpressPrice['freeExpressPrice'] : 0;
  3232. //组合套餐是否计算运费
  3233. if ($details['expressType'] = StatusCode::$delete){
  3234. //计算运费
  3235. $result = self::calExpress($c_goods_Lists);
  3236. if (!$result->isSuccess()) {
  3237. return ResultWrapper::fail($result->getData(), $result->getErrorCode());
  3238. }
  3239. }
  3240. $c_shop = [
  3241. 'shopId' => getArrayItem($shop, 'id', 0),
  3242. 'shopName' => getArrayItem($shop, 'name', ''),
  3243. 'shopLogo' => getArrayItem($shop, 'logo', ''),
  3244. 'expressMoney' => $this->expressMoney,
  3245. 'totalMoney' => floatval($details['price']),
  3246. 'preferential' => 0,
  3247. 'payMoney' => bcadd(floatval($details['price']),$this->expressMoney,2),
  3248. 'shopGoodsData' => $c_goods_Lists
  3249. ];
  3250. $ret = [
  3251. 'goodsData' => [$c_shop],
  3252. 'totalMoney' => floatval($details['price']),
  3253. 'payMoney' => bcadd(floatval($details['price']),$this->expressMoney,2),
  3254. 'preferential' => 0,
  3255. 'vipDiscount' => 0,
  3256. 'vipDoubleDiscount' => 0,
  3257. 'activityMoney' => 0,
  3258. 'expressMoney' => $this->expressMoney,
  3259. 'checkNum' => 0,
  3260. 'cartNum' => 0,
  3261. 'goodsNum' => 0,
  3262. 'invalidData' => [],
  3263. 'address' => $address,
  3264. 'useCoupon' => [],
  3265. 'payment' => $paymentData,
  3266. 'delivery' => $DeliveryData,
  3267. 'selfExpressRule' => $selfExpressRule,
  3268. 'coupon' => [],
  3269. 'vipCard' => [],
  3270. 'comBinId' => $data['comBinId']
  3271. ];
  3272. return ResultWrapper::success($ret);
  3273. }
  3274. }