// +---------------------------------------------------------------------- namespace app\services\activity\discounts; use app\dao\activity\discounts\StoreDiscountsDao; use app\services\BaseServices; use app\services\product\branch\StoreBranchProductServices; use app\services\product\product\StoreProductServices; use app\services\product\sku\StoreProductAttrResultServices; use app\services\product\sku\StoreProductAttrServices; use app\services\product\sku\StoreProductAttrValueServices; use app\services\user\label\UserLabelServices; use crmeb\exceptions\AdminException; use crmeb\services\CacheService; /** * 优惠套餐 * Class StoreDiscountsServices * @package app\services\activity\discounts * @mixin StoreDiscountsDao */ class StoreDiscountsServices extends BaseServices { /** * 商品活动类型 */ const TYPE = 5; /** * @param StoreDiscountsDao $dao */ public function __construct(StoreDiscountsDao $dao) { $this->dao = $dao; } /** * 保存数据 * @param $data * @return mixed */ public function saveDiscounts($data) { if ($data['freight'] == 2 && !$data['postage']) { throw new AdminException('请设置运费金额'); } if ($data['freight'] == 3 && !$data['temp_id']) { throw new AdminException('请选择运费模版'); } /** @var StoreProductAttrServices $storeProductAttrServices */ $storeProductAttrServices = app()->make(StoreProductAttrServices::class); /** @var StoreDiscountsProductsServices $storeDiscountsProductsServices */ $storeDiscountsProductsServices = app()->make(StoreDiscountsProductsServices::class); return $this->transaction(function () use ($data, $storeProductAttrServices, $storeDiscountsProductsServices) { //添加优惠套餐 $discountsData['title'] = $data['title']; $discountsData['image'] = $data['image']; $discountsData['vip_type'] = $data['vip_type']; $discountsData['type'] = $data['type']; $discountsData['is_limit'] = $data['is_limit']; $discountsData['limit_num'] = $data['is_limit'] ? $data['limit_num'] : 0; $discountsData['link_ids'] = implode(',', $data['link_ids']); $discountsData['is_time'] = $data['is_time']; $discountsData['start_time'] = $data['is_time'] ? strtotime($data['time'][0]) : 0; $discountsData['stop_time'] = $data['is_time'] ? strtotime($data['time'][1]) + 86399 : 0; $discountsData['sort'] = $data['sort']; $discountsData['add_time'] = time(); $discountsData['free_shipping'] = $data['free_shipping']; $discountsData['status'] = $data['status']; $discountsData['freight'] = $data['freight']; $discountsData['postage'] = $data['postage']; $discountsData['custom_form'] = $data['custom_form'] ? json_encode($data['custom_form']) : ''; $discountsData['system_form_id'] = $data['system_form_id'] ?? 0; $product_ids = []; foreach ($data['products'] as $product) { if ($product['type']) { $product_ids = []; $product_ids[] = $product['product_id']; break; } else { $product_ids[] = $product['product_id']; } } $discountsData['product_ids'] = implode(',', $product_ids); if ($data['id']) { unset($discountsData['add_time']); $this->dao->update($data['id'], $discountsData); $storeDiscountsProductsServices->del($data['id']); $discountsId = $data['id']; } else { $discountsId = $this->dao->save($discountsData)->id; } if (!$discountsId) throw new AdminException('添加失败'); if ($discountsData['is_limit']) CacheService::setStock(md5($discountsId), (int)$discountsData['limit_num'], 5); //添加优惠套餐内商品 /** @var StoreProductServices $productService */ $productService = app()->make(StoreProductServices::class); foreach ($data['products'] as $item) { $stock = $productService->value(['id' => $item['product_id'], 'is_del' => 0, 'is_show' => 1], 'stock') ?? 0; if ((int)$stock <= 0) { throw new AdminException('商品库存不足,请选择其他商品'); } $productData = []; $productData['discount_id'] = $discountsId; $productData['product_id'] = $item['product_id']; $productData['product_type'] = $item['product_type'] ?? 0; $productData['title'] = $item['store_name']; $productData['image'] = $item['image']; $productData['type'] = $item['type']; $productData['temp_id'] = $item['temp_id']; $discountsProducts = $storeDiscountsProductsServices->save($productData); $storeProductAttrServices->setItem('store_product_id', $item['product_id']); $skuList = $storeProductAttrServices->validateProductAttr($item['items'], $item['attr'], (int)$discountsProducts->id, 5); $storeProductAttrServices->reset(); $valueGroup = $storeProductAttrServices->saveProductAttr($skuList, (int)$discountsProducts->id, 5); if (!$discountsProducts || !$valueGroup) throw new AdminException('添加失败'); } return true; }); } /** * 获取列表 * @param array $where * @return array */ public function getList($where = []) { [$page, $limit] = $this->getPageValue(); $list = $this->dao->getList($where, ['products'], $page, $limit); $count = $this->dao->count($where + ['is_del' => 0]); $time = time(); //查询商品属性 /** @var StoreProductAttrValueServices $attrValueServices */ $attrValueServices = app()->make(StoreProductAttrValueServices::class); foreach ($list as &$item) { if (!$this->checkDiscount($item, 0) || ($item['stop_time'] && $item['stop_time'] < $time)) { $item['status'] = 0; $this->dao->update(['id' => $item['id']], ['status' => 0]); } $item['ot_price'] = 0; $item['price'] = 0; foreach($item['products'] as $key => &$value){ $attrInfo = $attrValueServices->getOne(['product_id' => $value['id'], 'type' => 5]); $item['ot_price'] = bcadd($item['ot_price'],$attrInfo['ot_price']*$attrInfo['pnum']); $item['price'] = bcadd($item['price'],$attrInfo['price']*$attrInfo['pnum']); $value['pnum'] = $attrInfo['pnum']; } $item['start_time'] = $item['start_time'] ? date('Y-m-d', $item['start_time']) : 0; $item['stop_time'] = $item['stop_time'] ? date('Y-m-d', $item['stop_time']) : 0; $item['add_time'] = date('Y-m-d H:i:s', $item['add_time']); } return compact('list', 'count'); } /** * 获取详情 * @param $id * @return array * @throws \think\db\exception\DataNotFoundException * @throws \think\db\exception\DbException * @throws \think\db\exception\ModelNotFoundException */ public function getInfo($id) { $discounts = $this->dao->get($id); if ($discounts) { $discounts = $discounts->toArray(); } else { throw new AdminException('套餐商品未找到!'); } /** @var StoreDiscountsProductsServices $discountsProducts */ $discountsProducts = app()->make(StoreDiscountsProductsServices::class); /** @var UserLabelServices $userLabelServices */ $userLabelServices = app()->make(UserLabelServices::class); $discounts['products'] = $discountsProducts->dao->getList(['discount_id' => $discounts['id']]); foreach ($discounts['products'] as &$item) { /** @var StoreProductAttrResultServices $storeProductAttrResultServices */ $storeProductAttrResultServices = app()->make(StoreProductAttrResultServices::class); $discountsResult = $storeProductAttrResultServices->value(['product_id' => $item['id'], 'type' => 5], 'result'); $item['items'] = json_decode($discountsResult, true)['attr']; $item['attr'] = $this->getattr($item['items'], $item['id'], 5); $item['store_name'] = $item['title']; } if ($discounts['start_time']) { $discounts['time'] = [date('Y-m-d', $discounts['start_time']), date('Y-m-d', $discounts['stop_time'])]; } else { $discounts['time'] = []; } $link_ids = explode(',', $discounts['link_ids']); $discounts['link_ids'] = $userLabelServices->getLabelList(['ids' => $link_ids], ['id', 'label_name']); return $discounts; } /** * 获取规格 * @param $attr * @param $id * @param $type * @return array */ public function getattr($attr, $id, $type) { /** @var StoreProductAttrValueServices $storeProductAttrValueServices */ $storeProductAttrValueServices = app()->make(StoreProductAttrValueServices::class); $value = attr_format($attr)[1]; $valueNew = []; $count = 0; foreach ($value as $key => $item) { $detail = $item['detail']; $suk = implode(',', $item['detail']); $sukValue = $storeProductAttrValueServices->getSkuArray(['product_id' => $id, 'type' => $type, 'suk' => $suk], 'bar_code,code,cost,price,ot_price,stock,image as pic,weight,volume,brokerage,brokerage_two,quota,quota_show,pnum', 'suk'); if (count($sukValue)) { $valueNew[$count]['value'] = ''; foreach (array_values($detail) as $k => $v) { $valueNew[$count]['value' . ($k + 1)] = $v; $valueNew[$count]['value'] .= $valueNew[$count]['value'] == '' ? $v : ',' . $v; } $valueNew[$count]['detail'] = $detail; $valueNew[$count]['pic'] = $sukValue[$suk]['pic'] ?? ''; $valueNew[$count]['price'] = $sukValue[$suk]['price'] ? floatval($sukValue[$suk]['price']) : 0; $valueNew[$count]['cost'] = $sukValue[$suk]['cost'] ? floatval($sukValue[$suk]['cost']) : 0; $valueNew[$count]['ot_price'] = isset($sukValue[$suk]['ot_price']) ? floatval($sukValue[$suk]['ot_price']) : 0; $valueNew[$count]['stock'] = $sukValue[$suk]['stock'] ? intval($sukValue[$suk]['stock']) : 0; // $valueNew[$count]['quota'] = $sukValue[$suk]['quota'] ? intval($sukValue[$suk]['quota']) : 0; $valueNew[$count]['quota'] = isset($sukValue[$suk]['quota_show']) && $sukValue[$suk]['quota_show'] ? intval($sukValue[$suk]['quota_show']) : 0; $valueNew[$count]['bar_code'] = $sukValue[$suk]['bar_code'] ?? ''; $valueNew[$count]['code'] = $sukValue[$suk]['code'] ?? ''; $valueNew[$count]['pnum'] = $sukValue[$suk]['pnum'] ?? 0; $valueNew[$count]['p_price'] = floatval($valueNew[$count]['pnum'] * $valueNew[$count]['price']); $valueNew[$count]['weight'] = $sukValue[$suk]['weight'] ? floatval($sukValue[$suk]['weight']) : 0; $valueNew[$count]['volume'] = $sukValue[$suk]['volume'] ? floatval($sukValue[$suk]['volume']) : 0; $valueNew[$count]['brokerage'] = $sukValue[$suk]['brokerage'] ? floatval($sukValue[$suk]['brokerage']) : 0; $valueNew[$count]['brokerage_two'] = $sukValue[$suk]['brokerage_two'] ? floatval($sukValue[$suk]['brokerage_two']) : 0; $count++; } } return $valueNew; } /** * 修改状态 * @param $id * @param $status * @return mixed */ public function setStatus($id, $status) { $discounts = $this->dao->get($id, ['*'], ['products']); if ($discounts) { $discounts = $discounts->toArray(); } else { throw new AdminException('套餐商品未找到!'); } //上架 if ($status) { if ($discounts['stop_time'] && $discounts['stop_time'] < time()) { throw new AdminException('套餐活动已结束'); } if (!$this->checkDiscount($discounts, 0)) { throw new AdminException('套餐内商品已下架或者库存不足'); } } return $this->dao->update($id, ['status' => $status]); } /** * 删除商品 * @param $id * @return mixed */ public function del($id) { return $this->dao->update($id, ['is_del' => 1]); } /** * 获取优惠套餐列表 * @param int $product_id * @param int $uid * @param int $limit * @param string $field * @return mixed * @throws \think\db\exception\DataNotFoundException * @throws \think\db\exception\DbException * @throws \think\db\exception\ModelNotFoundException */ public function getDiscounts(int $product_id, int $uid, int $limit = 0, string $field = '*') { if ($limit) { $page = 0; } else { [$page, $limit] = $this->getPageValue(); } $list = $this->dao->getDiscounts($product_id, $field, $page, $limit); foreach ($list as $key => &$discounts) { $discounts = $this->checkDiscount($discounts, $uid); if (!$discounts) unset($list[$key]); $maxDiscountsPrice = 0; if (isset($discounts['products']) && $discounts['products']) { $oneProductMax = []; foreach ($discounts['products'] as $product) { $attrInfo = $product['productValue'] ?? []; if ($attrInfo) { $one = 0; foreach ($attrInfo as $attr) { $one = max($one, (float)bcsub((string)($attr['product_price'] ?? 0), (string)($attr['price'] ?? 0), 2)); } $oneProductMax[] = $one; } } $maxDiscountsPrice = array_sum($oneProductMax); } $discounts['max_discounts_price'] = $maxDiscountsPrice; } return array_merge($list); } /** * 购买|退款处理套餐限量 * @param int $discountId * @param bool $is_dec * @return bool */ public function changeDiscountLimit(int $discountId, bool $is_dec = true) { $is_limit = $this->dao->value(['id' => $discountId], 'is_limit'); $res = true; //开启限量 if ($is_limit) { if ($is_dec) $res = $res && $this->dao->decLimitNum($discountId); else $res = $res && $this->dao->incLimitNum($discountId); } return $res; } /** * 优惠套餐库存减少 * @param int $num * @param int $discountId * @param int $discount_product_id * @param int $product_id * @param string $unique * @param int $store_id * @return bool */ public function decDiscountStock(int $num, int $discountId, int $discount_product_id, int $product_id, string $unique, int $store_id = 0) { if (!$discountId || !$discount_product_id || !$product_id) return false; $res = true; if ($unique) { /** @var StoreProductAttrValueServices $skuValueServices */ $skuValueServices = app()->make(StoreProductAttrValueServices::class); //套餐商品sku $suk = $skuValueServices->value(['unique' => $unique, 'product_id' => $discount_product_id, 'type' => 5], 'suk'); //平台商品sku unique $productUnique = $skuValueServices->value(['suk' => $suk, 'product_id' => $product_id, 'type' => 0], 'unique'); /** @var StoreProductServices $services */ $services = app()->make(StoreProductServices::class); //减掉普通商品sku的库存加销量 $res = false !== $services->decProductStock($num, $product_id, (string)$productUnique, $store_id); } return $res; } /** * 加库存减销量 * @param int $num * @param int $discountId 套餐id * @param int $discount_product_id 套餐商品id * @param int $product_id 原商品ID * @param string $unique * @return bool */ public function incDiscountStock(int $num, int $discountId, int $discount_product_id, int $product_id, string $unique, int $store_id = 0) { if (!$discountId || !$discount_product_id || !$product_id) return false; $res = true; if ($unique) { /** @var StoreProductAttrValueServices $skuValueServices */ $skuValueServices = app()->make(StoreProductAttrValueServices::class); //套餐商品sku $suk = $skuValueServices->value(['unique' => $unique, 'product_id' => $discount_product_id, 'type' => 5], 'suk'); //平台普通商品sku unique $productUnique = $skuValueServices->value(['suk' => $suk, 'product_id' => $product_id, 'type' => 0], 'unique'); /** @var StoreProductServices $services */ $services = app()->make(StoreProductServices::class); //增加当前普通商品sku的库存,减去销量 $res = $res && $services->incProductStock($num, $product_id, (string)$productUnique, $store_id); } return $res; } /** * 判断优惠套餐显示 * @param $discount * @param $uid * @return false */ public function checkDiscount($discount, $uid) { $discount = is_object($discount) ? $discount->toArray() : $discount; if ($discount['is_limit'] && $discount['limit_num'] <= 0) return false; if (isset($discount['products']) && $discount['products']) { /** @var StoreProductServices $productService */ $productService = app()->make(StoreProductServices::class); /** @var StoreProductAttrServices $storeProductAttrServices */ $storeProductAttrServices = app()->make(StoreProductAttrServices::class); $minPrice = 0; foreach ($discount['products'] as $key => &$item) { $stock = $productService->value(['id' => $item['product_id'], 'is_del' => 0, 'is_show' => 1], 'stock') ?? 0; try { [$productAttr, $productValue] = $storeProductAttrServices->getProductAttrDetailCache($item['id'], $uid, 0, 5, $item['product_id']); } catch(\Throwable $e) { return false; } if ($discount['type']) { if ($item['type']) { if ($stock == 0) { return false; } else { if (!array_sum(array_column($productValue, 'product_stock'))) return false; } } else { if (!array_sum(array_column($productValue, 'product_stock'))) unset($discount['products'][$key]); } } else { if ($stock == 0) { return false; } else { if (!array_sum(array_column($productValue, 'product_stock'))) return false; } } $item['productAttr'] = $productAttr; $item['productValue'] = $productValue; $minPrice += (min(array_column($productValue, 'price')) ?? 0); } if (!count($discount['products'])) return false; $discount['min_price'] = $minPrice; $discount['products'] = array_merge($discount['products']); } return $discount; } }