You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
356 lines
13 KiB
356 lines
13 KiB
3 months ago
|
<?php
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Copyright (c) 2016~2020 https://www.crmeb.com All rights reserved.
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||
|
// +----------------------------------------------------------------------
|
||
|
// | Author: CRMEB Team <admin@crmeb.com>
|
||
|
// +----------------------------------------------------------------------
|
||
|
declare (strict_types=1);
|
||
|
|
||
|
namespace app\services\activity;
|
||
|
|
||
|
|
||
|
use app\dao\activity\StoreActivityDao;
|
||
|
use app\services\activity\seckill\StoreSeckillServices;
|
||
|
use app\services\activity\seckill\StoreSeckillTimeServices;
|
||
|
use app\services\BaseServices;
|
||
|
use app\services\product\product\StoreDescriptionServices;
|
||
|
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\store\SystemStoreServices;
|
||
|
use crmeb\exceptions\AdminException;
|
||
|
use crmeb\services\CacheService;
|
||
|
|
||
|
/**
|
||
|
* 活动
|
||
|
* Class StoreActivityServices
|
||
|
* @package app\services\activity
|
||
|
* @mixin StoreActivityDao
|
||
|
*/
|
||
|
class StoreActivityServices extends BaseServices
|
||
|
{
|
||
|
|
||
|
/**
|
||
|
* @var string[]
|
||
|
*/
|
||
|
public $typeName = [
|
||
|
1 => '秒杀',
|
||
|
];
|
||
|
|
||
|
/**
|
||
|
* StoreActivityServices constructor.
|
||
|
* @param StoreActivityDao $dao
|
||
|
*/
|
||
|
public function __construct(StoreActivityDao $dao)
|
||
|
{
|
||
|
$this->dao = $dao;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 活动列表
|
||
|
* @param array $where
|
||
|
* @return array
|
||
|
* @throws \think\db\exception\DataNotFoundException
|
||
|
* @throws \think\db\exception\DbException
|
||
|
* @throws \think\db\exception\ModelNotFoundException
|
||
|
*/
|
||
|
public function systemPage(array $where)
|
||
|
{
|
||
|
[$page, $limit] = $this->getPageValue();
|
||
|
$list = $this->dao->getList($where, '*', $page, $limit, ['seckill' => function ($query) {
|
||
|
$query->field('id,activity_id');
|
||
|
}]);
|
||
|
$count = $this->dao->count($where);
|
||
|
/** @var StoreSeckillTimeServices $seckillTimeServices */
|
||
|
$seckillTimeServices = app()->make(StoreSeckillTimeServices::class);
|
||
|
$timeList = $seckillTimeServices->time_list();
|
||
|
if ($timeList) {
|
||
|
$timeList = array_combine(array_column($timeList, 'id'), $timeList);
|
||
|
}
|
||
|
foreach ($list as &$item) {
|
||
|
$item['product_count'] = count($item['seckill']);
|
||
|
$item['time_id'] = is_string($item['time_id']) ? explode(',', $item['time_id']) : $item['time_id'];
|
||
|
$item['time_list'] = [];
|
||
|
foreach ($item['time_id'] as $time_id) {
|
||
|
if (isset($timeList[$time_id])) $item['time_list'][] = $timeList[$time_id];
|
||
|
}
|
||
|
if ($item['status']) {
|
||
|
if ($item['start_day'] > time())
|
||
|
$item['start_name'] = '未开始';
|
||
|
else if (bcadd((string)$item['end_day'], '86400') < time())
|
||
|
$item['start_name'] = '已结束';
|
||
|
else if (bcadd((string)$item['end_day'], '86400') > time() && $item['start_day'] < time()) {
|
||
|
$item['start_name'] = '进行中';
|
||
|
}
|
||
|
} else $item['start_name'] = '已结束';
|
||
|
switch ($item['type']) {
|
||
|
case 1://秒杀
|
||
|
$item['start_time'] = $item['start_time'] ? substr_replace($item['start_time'], ':', 2, 0) : '';
|
||
|
$item['end_time'] = $item['start_time'] ? substr_replace($item['end_time'], ':', 2, 0) : '';
|
||
|
break;
|
||
|
}
|
||
|
$item['start_day'] = date('Y-m-d', $item['start_day']);
|
||
|
$item['end_day'] = date('Y-m-d', $item['end_day']);
|
||
|
// $item['add_time'] = $item['add_time'] ? date('Y-m-d H:i:s', $item['add_time']) : '';
|
||
|
}
|
||
|
return compact('list', 'count');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取一条活动信息
|
||
|
* @param int $id
|
||
|
* @param array $field
|
||
|
* @return array
|
||
|
* @throws \think\db\exception\DataNotFoundException
|
||
|
* @throws \think\db\exception\DbException
|
||
|
* @throws \think\db\exception\ModelNotFoundException
|
||
|
*/
|
||
|
public function getInfo(int $id, array $field = ['*'])
|
||
|
{
|
||
|
$info = $this->dao->get($id, $field);
|
||
|
if (!$info) {
|
||
|
throw new AdminException('数据不存在');
|
||
|
}
|
||
|
$info = $info->toArray();
|
||
|
return $info;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取一条秒杀活动
|
||
|
* @param int $id
|
||
|
* @return array
|
||
|
* @throws \think\db\exception\DataNotFoundException
|
||
|
* @throws \think\db\exception\DbException
|
||
|
* @throws \think\db\exception\ModelNotFoundException
|
||
|
*/
|
||
|
public function getSeckillInfo(int $id)
|
||
|
{
|
||
|
$info = $this->getInfo($id);
|
||
|
/** @var StoreSeckillServices $seckillServices */
|
||
|
$seckillServices = app()->make(StoreSeckillServices::class);
|
||
|
$seckill = $seckillServices->getList(['activity_id' => $id, 'is_del' => 0], 0, 0, ['attrValue']);
|
||
|
$info['section_data'] = [date('Y-m-d', $info['start_day']), date('Y-m-d', $info['end_day'])];
|
||
|
$productList = [];
|
||
|
if ($seckill) {
|
||
|
/** @var StoreProductServices $productServices */
|
||
|
$productServices = app()->make(StoreProductServices::class);
|
||
|
$productList = $productServices->searchList(['id' => array_column($seckill, 'product_id'), 'is_del' => 0, 'is_verify' => 1]);
|
||
|
$productList = $productList['list'] ?? [];
|
||
|
$seckill = array_combine(array_column($seckill, 'product_id'), $seckill);
|
||
|
//放入秒杀商品价格
|
||
|
foreach ($productList as &$product) {
|
||
|
$seckillInfo = $seckill[$product['id']] ?? [];
|
||
|
$attrValue = $product['attrValue'] ?? [];
|
||
|
if ($seckillInfo && $attrValue) {
|
||
|
$product['status'] = $seckillInfo['status'];
|
||
|
$seckillAttrValue = $seckillInfo['attrValue'] ?? [];
|
||
|
if ($seckillAttrValue) {
|
||
|
$seckillAttrValue = array_combine(array_column($seckillAttrValue, 'suk'), $seckillAttrValue);
|
||
|
foreach ($attrValue as &$value) {
|
||
|
$value['quota'] = $seckillAttrValue[$value['suk']]['quota'] ?? 0;
|
||
|
$value['quota_show'] = $seckillAttrValue[$value['suk']]['quota_show'] ?? 0;
|
||
|
$value['price'] = $seckillAttrValue[$value['suk']]['price'] ?? 0;
|
||
|
$value['cost'] = $seckillAttrValue[$value['suk']]['cost'] ?? 0;
|
||
|
$value['ot_price'] = $seckillAttrValue[$value['suk']]['ot_price'] ?? 0;
|
||
|
}
|
||
|
$product['attrValue'] = $attrValue;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
$info['productList'] = $productList;
|
||
|
//适用门店
|
||
|
$info['stores'] = [];
|
||
|
if (isset($info['applicable_type']) && ($info['applicable_type'] == 1 || ($info['applicable_type'] == 2 && isset($info['applicable_store_id']) && $info['applicable_store_id']))) {//查询门店信息
|
||
|
$where = ['is_del' => 0];
|
||
|
if ($info['applicable_type'] == 2) {
|
||
|
$store_ids = is_array($info['applicable_store_id']) ? $info['applicable_store_id'] : explode(',', $info['applicable_store_id']);
|
||
|
$where['id'] = $store_ids;
|
||
|
}
|
||
|
$field = ['id', 'cate_id', 'name', 'phone', 'address', 'detailed_address', 'image', 'is_show', 'day_time', 'day_start', 'day_end'];
|
||
|
/** @var SystemStoreServices $storeServices */
|
||
|
$storeServices = app()->make(SystemStoreServices::class);
|
||
|
$storeData = $storeServices->getStoreList($where, $field, '', '', 0, ['categoryName']);
|
||
|
$info['stores'] = $storeData['list'] ?? [];
|
||
|
}
|
||
|
return $info;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* 保存活动信息
|
||
|
* @param int $id
|
||
|
* @param array $data
|
||
|
* @param array $expend
|
||
|
* @param int $type
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function saveData(int $id, array $data, array $expend, int $type = 1)
|
||
|
{
|
||
|
if (!$data || !$expend) {
|
||
|
throw new AdminException('缺少活动数据');
|
||
|
}
|
||
|
$section_data = $data['section_data'];
|
||
|
$data['start_day'] = strtotime($section_data[0]);
|
||
|
$data['end_day'] = strtotime($section_data[1]);
|
||
|
unset($data['section_data']);
|
||
|
$this->transaction( function () use ($id, $type, $data, $section_data, $expend) {
|
||
|
if ($id) {
|
||
|
$this->getInfo($id);
|
||
|
$this->dao->update($id, $data);
|
||
|
$this->clearAcivityAttr($id, $type);
|
||
|
} else {
|
||
|
$data['add_time'] = time();
|
||
|
$res = $this->dao->save($data);
|
||
|
$id = (int)$res->id;
|
||
|
}
|
||
|
//处理活动商品
|
||
|
switch ($type) {
|
||
|
case 1:
|
||
|
$data['section_data'] = $section_data;
|
||
|
$this->saveActivitySeckill($id, $expend, $data);
|
||
|
break;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 保存秒杀商品
|
||
|
* @param int $id
|
||
|
* @param array $data
|
||
|
* @param array $activity_data
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function saveActivitySeckill(int $id, array $data, array $activity_data)
|
||
|
{
|
||
|
$productIds = array_column($data, 'id');
|
||
|
/** @var StoreProductServices $productServices */
|
||
|
$productServices = app()->make(StoreProductServices::class);
|
||
|
$productList = $productServices->searchList(['id' => $productIds, 'is_del' => 0, 'is_verify' => 1]);
|
||
|
$productList = $productList['list'] ?? [];
|
||
|
//放入秒杀商品价格
|
||
|
/** @var StoreSeckillServices $seckillServices */
|
||
|
$seckillServices = app()->make(StoreSeckillServices::class);
|
||
|
$data = array_combine($productIds, $data);
|
||
|
foreach ($productList as &$product) {
|
||
|
$attrInfo = $productServices->getProductRules((int)$product['id'], 0);
|
||
|
$seckillData = [];
|
||
|
$seckillData['activity_id'] = $id;
|
||
|
$seckillData['product_id'] = $product['id'] ?? 0;
|
||
|
$seckillData['title'] = $product['store_name'] ?? '';
|
||
|
$seckillData['info'] = $product['store_info'] ?? '';
|
||
|
$seckillData['unit_name'] = $product['unit_name'] ?? '';
|
||
|
$seckillData['section_time'] = $activity_data['section_data'];
|
||
|
$seckillData['images'] = $product['slider_image'] ?? '';
|
||
|
$seckillData['description'] = $product['description'] ?? '';
|
||
|
$seckillData['status'] = $data[$product['id']]['status'] ?? 1;
|
||
|
$seckillData['time_id'] = $activity_data['time_id'] ?? [];
|
||
|
$seckillData['num'] = $activity_data['num'] ?? 0;
|
||
|
$seckillData['once_num'] = $activity_data['once_num'] ?? 0;
|
||
|
$seckillData['delivery_type'] = $product['delivery_type'] ?? '';
|
||
|
$seckillData['temp_id'] = $product['temp_id'];//运费设置
|
||
|
$seckillData['freight'] = $product['freight'];//运费设置
|
||
|
$seckillData['postage'] = $product['postage'];//邮费
|
||
|
$seckillData['custom_form'] = $product['custom_form'];//自定义表单
|
||
|
$seckillData['system_form_id'] = $product['system_form_id'];//系统表单ID
|
||
|
$seckillData['product_type'] = $product['product_type'];//商品类型
|
||
|
$seckillData['applicable_type'] = $activity_data['applicable_type'];//适用门店类型
|
||
|
$seckillData['applicable_store_id'] = $activity_data['applicable_store_id'];//适用门店IDS
|
||
|
|
||
|
$seckillData['items'] = $attrInfo['items'];
|
||
|
$attrs = $attrInfo['attrs'] ?? [];
|
||
|
if ($attrs) {
|
||
|
$seckillAttrValue = $data[$product['id']]['attrValue'] ?? [];
|
||
|
if (!$seckillAttrValue) {
|
||
|
throw new AdminException('请选择商品规格');
|
||
|
}
|
||
|
foreach ($seckillAttrValue as $sattr) {
|
||
|
if (!isset($sattr['price']) || !$sattr['price']) {
|
||
|
throw new AdminException('请填写商品('.$product['store_name'] .')活动价');
|
||
|
}
|
||
|
if (!isset($sattr['quota']) || !$sattr['quota']) {
|
||
|
throw new AdminException('请填写商品('.$product['store_name'] .')限量');
|
||
|
}
|
||
|
}
|
||
|
$seckillAttrValue = array_combine(array_column($seckillAttrValue, 'suk'), $seckillAttrValue);
|
||
|
foreach ($attrs as $attr) {
|
||
|
$sku = implode(',', $attr['detail']);
|
||
|
if(!isset($seckillAttrValue[$sku])) {
|
||
|
throw new AdminException('请重新选择商品规格');
|
||
|
}
|
||
|
if (($seckillAttrValue[$sku][''] ?? 0) > $attr['stock']) {
|
||
|
throw new AdminException('限量超过了商品库存');
|
||
|
}
|
||
|
$attr['quota'] = $attr['quota_show'] = $seckillAttrValue[$sku]['quota'] ?? 0;
|
||
|
$attr['price'] = $seckillAttrValue[$sku]['price'] ?? 0;
|
||
|
$attr['cost'] = $seckillAttrValue[$sku]['cost'] ?? 0;
|
||
|
$attr['ot_price'] = $seckillAttrValue[$sku]['ot_price'] ?? 0;
|
||
|
$seckillData['attrs'][] = $attr;
|
||
|
}
|
||
|
}
|
||
|
$seckillServices->saveData(0, $seckillData);
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 清空活动商品数据
|
||
|
* @param int $id
|
||
|
* @return bool
|
||
|
* @throws \think\db\exception\DataNotFoundException
|
||
|
* @throws \think\db\exception\DbException
|
||
|
* @throws \think\db\exception\ModelNotFoundException
|
||
|
*/
|
||
|
public function clearAcivityAttr(int $id, int $type = 1)
|
||
|
{
|
||
|
/** @var StoreSeckillServices $seckillServices */
|
||
|
$seckillServices = app()->make(StoreSeckillServices::class);
|
||
|
$seckill = $seckillServices->getList(['activity_id' => $id, 'is_del' => 0]);
|
||
|
if ($seckill) {
|
||
|
$seckillIds = array_column($seckill, 'id');
|
||
|
/** @var StoreProductAttrResultServices $storeProductAttrResultServices */
|
||
|
$storeProductAttrResultServices = app()->make(StoreProductAttrResultServices::class);
|
||
|
/** @var StoreDescriptionServices $storeDescriptionServices */
|
||
|
$storeDescriptionServices = app()->make(StoreDescriptionServices::class);
|
||
|
/** @var StoreProductAttrServices $storeProductAttrServices */
|
||
|
$storeProductAttrServices = app()->make(StoreProductAttrServices::class);
|
||
|
/** @var StoreProductAttrValueServices $storeProductAttrValueServices */
|
||
|
$storeProductAttrValueServices = app()->make(StoreProductAttrValueServices::class);
|
||
|
$where = ['product_id' => $seckillIds, 'type' => $type];
|
||
|
$storeProductAttrResultServices->delete($where);
|
||
|
$storeDescriptionServices->delete($where);
|
||
|
$storeProductAttrServices->delete($where);
|
||
|
$storeProductAttrValueServices->delete($where);
|
||
|
$seckillServices->delete(['activity_id' => $id, 'is_del' => 0]);
|
||
|
$seckillServices->cacheTag()->clear();
|
||
|
}
|
||
|
CacheService::redisHandler('product_attr')->clear();
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* 判断时间段是否存在秒杀活动
|
||
|
* @param array $where
|
||
|
* @param int $type
|
||
|
* @return bool
|
||
|
* @throws \think\db\exception\DbException
|
||
|
*/
|
||
|
public function existenceActivity(array $where, int $type = 1): bool
|
||
|
{
|
||
|
$where['is_del'] = 0;
|
||
|
if (!isset($where['type'])) $where['type'] = $type;
|
||
|
return $this->dao->checkActivity($where);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
}
|