| 
									
										
										
										
											2022-05-25 19:35:57 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace app\traits\order; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use app\exception\RepositoryException; | 
					
						
							|  |  |  | use app\model\mall\SpuLimitTime; | 
					
						
							|  |  |  | use app\model\OrderSku; | 
					
						
							|  |  |  | use app\model\ShoppingCart; | 
					
						
							|  |  |  | use app\model\Sku; | 
					
						
							|  |  |  | use app\model\Spu; | 
					
						
							|  |  |  | use app\repository\OrderRepository; | 
					
						
							|  |  |  | use app\repository\SpuRepository; | 
					
						
							|  |  |  | use Exception; | 
					
						
							|  |  |  | use think\Collection; | 
					
						
							|  |  |  | use think\db\exception\DataNotFoundException; | 
					
						
							|  |  |  | use think\db\exception\DbException; | 
					
						
							|  |  |  | use think\db\exception\ModelNotFoundException; | 
					
						
							|  |  |  | use think\facade\Db; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * 购物车相关 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Trait ShoppingCartTrait | 
					
						
							|  |  |  |  * @package app\traits\order | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | trait ShoppingCartTrait | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 获取购物车列表 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param  int  $accountId | 
					
						
							|  |  |  |      * @param  string  $type | 
					
						
							|  |  |  |      * @param  int  $page | 
					
						
							|  |  |  |      * @param  int  $size | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      * @throws DataNotFoundException | 
					
						
							|  |  |  |      * @throws DbException | 
					
						
							|  |  |  |      * @throws ModelNotFoundException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function shoppingCart(int $accountId, string $type = '', int $page = 1, int $size = 20): array | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $isScore = $type == 'score' ? 1 : 0; | 
					
						
							|  |  |  |         $res     = [ | 
					
						
							|  |  |  |             'total'   => 0, | 
					
						
							|  |  |  |             'current' => $page, | 
					
						
							|  |  |  |             'size'    => $size, | 
					
						
							|  |  |  |             'list'    => new Collection(), | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $query        = ShoppingCart::where('account_id', $accountId) | 
					
						
							|  |  |  |             ->where('is_score', $isScore) | 
					
						
							|  |  |  |             ->with([ | 
					
						
							|  |  |  |                 'spu'              => function ($query) { | 
					
						
							| 
									
										
										
										
											2022-05-27 13:55:45 +08:00
										 |  |  |                     $query->field('id,name as spu_name,cover as spu_cover,original_price,unit'); | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 'sku'           => function ($query) { | 
					
						
							| 
									
										
										
										
											2022-05-25 19:35:57 +08:00
										 |  |  |                     $query->field('id,title as sku_name,main_image as sku_cover,price as sku_price,coding,spec_text,type as activity_type'); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ]); | 
					
						
							|  |  |  |         $res['total'] = $query->count(); | 
					
						
							|  |  |  |         if ($res['total'] > 0) { | 
					
						
							|  |  |  |             $list = $query->order('updated_at', 'desc') | 
					
						
							|  |  |  |                 ->page($page) | 
					
						
							|  |  |  |                 ->limit($size) | 
					
						
							|  |  |  |                 ->select(); | 
					
						
							|  |  |  |             $res['list'] = $list->each(function ($item) { | 
					
						
							|  |  |  |                 if (!isset($item->sku) || !isset($item->sku->activity_type)) { | 
					
						
							|  |  |  |                     unset($item); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     if (!isset($item->sku->spec_text)) { | 
					
						
							|  |  |  |                         $item->sku->spec_text = []; | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         $specArr = json_decode($item->sku->spec_text, true); | 
					
						
							|  |  |  |                         $item->sku->spec_text = $specArr; | 
					
						
							|  |  |  |                         $item->sku->spec_info = SpuRepository::getInstance()->convertSpecInfo($specArr); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 unset($item->spu_activity); | 
					
						
							|  |  |  |                 unset($item->activity_skus); | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $res; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 购物车保存 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param  int  $accountId | 
					
						
							|  |  |  |      * @param  int  $skuId | 
					
						
							|  |  |  |      * @param  int  $num  最终购物车数量 注意 不是新增的数量 而是最终保存的数量。若=0 则删除 | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      * @throws DataNotFoundException | 
					
						
							|  |  |  |      * @throws DbException | 
					
						
							|  |  |  |      * @throws ModelNotFoundException | 
					
						
							|  |  |  |      * @throws RepositoryException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function shoppingCartAdd(int $accountId, int $skuId, int $num = 1): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!$skuInfo = SpuRepository::getInstance()->infoBySkuId($skuId)) { | 
					
						
							|  |  |  |             throw new RepositoryException('商品不存在或已下架'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($item = ShoppingCart::where('account_id', $accountId)->where('sku_id', $skuId)->find()) { | 
					
						
							|  |  |  |             $update['updated_at'] = date('Y-m-d H:i:s'); | 
					
						
							|  |  |  |             $update['spu_name']   = $skuInfo['spu_name'] ?? ($skuInfo['spu_activity_name'] ?? ''); | 
					
						
							|  |  |  |             $update['sku_price']  = $skuInfo['sku_price']; | 
					
						
							|  |  |  |             $update['score']      = $skuInfo['sku_score']; | 
					
						
							|  |  |  |             $update['sku_name']   = $skuInfo['sku_name']; | 
					
						
							|  |  |  |             $update['sku_cover']  = $skuInfo['sku_cover']; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $item->inc('num', $num)->update($update); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ShoppingCart::create([ | 
					
						
							|  |  |  |             'account_id'      => $accountId, | 
					
						
							|  |  |  |             'spu_activity_id' => $skuInfo['spu_activity_id'] ?? 0, | 
					
						
							|  |  |  |             'spu_id'          => $skuInfo['spu_id'], | 
					
						
							|  |  |  |             'sku_id'          => $skuInfo['id'], | 
					
						
							|  |  |  |             'original_price'  => $skuInfo['original_price'], | 
					
						
							|  |  |  |             'price'           => $skuInfo['price'], | 
					
						
							|  |  |  |             'score'           => $skuInfo['sku_score'], | 
					
						
							|  |  |  |             'num'             => $num, | 
					
						
							|  |  |  |             'created_at'      => date('Y-m-d H:i:s'), | 
					
						
							|  |  |  |             'spu_name'        => $skuInfo['spu_name'] ?? ($skuInfo['spu_activity_name'] ?? ''), | 
					
						
							|  |  |  |             'sku_price'       => $skuInfo['price'], | 
					
						
							|  |  |  |             'sku_name'        => $skuInfo['sku_name'], | 
					
						
							|  |  |  |             'sku_cover'       => $skuInfo['main_image'], | 
					
						
							|  |  |  |             'is_score'        => $skuInfo['is_score'] ?? 0, | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 购物车数量变更 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param  int  $id | 
					
						
							|  |  |  |      * @param  int  $num | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      * @throws DataNotFoundException | 
					
						
							|  |  |  |      * @throws DbException | 
					
						
							|  |  |  |      * @throws ModelNotFoundException | 
					
						
							|  |  |  |      * @throws RepositoryException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function shoppingCartChangeNum(int $id, int $num = 1): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!$item = ShoppingCart::where('id', $id)->find()) { | 
					
						
							|  |  |  |             throw new RepositoryException('记录不存在'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return $item->save(['num' => $num]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 购物车删除 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param  int  $accountId | 
					
						
							|  |  |  |      * @param  int  $id | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      * @throws DataNotFoundException | 
					
						
							|  |  |  |      * @throws DbException | 
					
						
							|  |  |  |      * @throws ModelNotFoundException | 
					
						
							|  |  |  |      * @throws RepositoryException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function shoppingCartDel(int $accountId, int $id): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!$item = ShoppingCart::where('id', $id)->find()) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($item['account_id'] != $accountId) { | 
					
						
							|  |  |  |             throw new RepositoryException('这不是您的购物车商品'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $item->delete(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 删除指定订单相关的购物车数据 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param  int  $accountId | 
					
						
							|  |  |  |      * @param  string  $orderId | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function delShoppingCartByOrder(int $accountId, string $orderId): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         Db::startTrans(); | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |             $orderSkus = OrderSku::where('coding', $orderId)->field('id,coding,sku_id,num')->select(); | 
					
						
							|  |  |  |             $sku2Num   = $orderSkus->column('num', 'sku_id'); | 
					
						
							|  |  |  |             if (empty($orderSkus)) { | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $list = ShoppingCart::where('account_id', $accountId) | 
					
						
							|  |  |  |                 ->field('id,sku_id,num') | 
					
						
							|  |  |  |                 ->whereIn('sku_id', $orderSkus->column('sku_id')) | 
					
						
							|  |  |  |                 ->lock(true)->select(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $update = []; | 
					
						
							|  |  |  |             foreach ($list as $v) { | 
					
						
							|  |  |  |                 $arr        = []; | 
					
						
							|  |  |  |                 $arr['num'] = $v['num'] - ($sku2Num[$v['sku_id']] ?? 1); | 
					
						
							|  |  |  |                 $arr['num'] = $arr['num'] < 0 ? 0 : $arr['num']; | 
					
						
							|  |  |  |                 $arr['id']  = $v['id']; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 $update[] = $arr; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             (new ShoppingCart())->saveAll($update); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             ShoppingCart::where('account_id', $accountId)->where('num', '<=', 0)->delete(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Db::commit(); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } catch (Exception $e) { | 
					
						
							|  |  |  |             Db::rollback(); | 
					
						
							|  |  |  |             OrderRepository::log('清理购物车数据失败', $e); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 检查并清理购物车不存在的商品和sku | 
					
						
							|  |  |  |      * @param  int  $accountId | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function checkShoppingCart(int $accountId = 0): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $where = []; | 
					
						
							|  |  |  |         if ($accountId > 0) { | 
					
						
							|  |  |  |             $where[] = ['account_id', '=', $accountId]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $items = ShoppingCart::where($where)->column('spu_id,sku_id'); | 
					
						
							|  |  |  |         $spuIds = array_unique(array_column($items, 'spu_id')); | 
					
						
							|  |  |  |         $skuIds = array_unique(array_column($items, 'sku_id')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $notExistSpu = []; | 
					
						
							|  |  |  |         if ($spuIds) { | 
					
						
							|  |  |  |             $existSpu = Spu::whereIn('id', $spuIds)->whereNull('deleted_at')->column('id'); | 
					
						
							|  |  |  |             $notExistSpu = array_diff($spuIds, $existSpu); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $notExistSku = []; | 
					
						
							|  |  |  |         if ($skuIds) { | 
					
						
							|  |  |  |             $existSku = Sku::whereIn('id', $skuIds)->whereNull('deleted_at')->column('id'); | 
					
						
							|  |  |  |             $notExistSku = array_diff($skuIds, $existSku); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($notExistSku) || !empty($notExistSpu)) { | 
					
						
							|  |  |  |             ShoppingCart::where($where) | 
					
						
							|  |  |  |                 ->where(function ($q) use ($notExistSku, $notExistSpu) { | 
					
						
							|  |  |  |                     if (!empty($notExistSpu)) { | 
					
						
							|  |  |  |                         $q->whereOr('spu_id', 'in', $notExistSpu); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     if (!empty($notExistSku)) { | 
					
						
							|  |  |  |                         $q->whereOr('sku_id', 'in', $notExistSku); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |                 ->delete(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |         //        return ['noteSpu' => $notExistSpu, 'notSku' => $notExistSku, 'needDel' => $needDel, 'count' => count($needDel)];
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |