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.
yanzong/app/store/model/Goods.php

312 lines
12 KiB

11 months ago
<?php
// +----------------------------------------------------------------------
// | 萤火商城系统 [ 致力于通过产品和服务,帮助商家高效化开拓市场 ]
// +----------------------------------------------------------------------
// | Copyright (c) 2017~2023 https://www.yiovo.com All rights reserved.
// +----------------------------------------------------------------------
// | Licensed 这不是一个自由软件,不允许对程序代码以任何形式任何目的的再发行
// +----------------------------------------------------------------------
// | Author: 萤火科技 <admin@yiovo.com>
// +----------------------------------------------------------------------
declare (strict_types=1);
namespace app\store\model;
use app\common\library\helper;
use app\store\model\Spec as SpecModel;
use app\common\model\Goods as GoodsModel;
use app\store\model\GoodsSku as GoodsSkuModel;
use app\store\model\GoodsImage as GoodsImageModel;
use app\store\model\GoodsSpecRel as GoodsSpecRelModel;
use app\store\model\goods\ServiceRel as GoodsServiceRelModel;
use app\store\model\GoodsCategoryRel as GoodsCategoryRelModel;
use app\store\service\Goods as GoodsService;
use app\common\enum\goods\SpecType as GoodsSpecTypeEnum;
use app\common\enum\goods\Status as GoodsStatusEnum;
use cores\exception\BaseException;
9 months ago
use app\common\model\Region;
11 months ago
/**
* 商品模型
* Class Goods
* @package app\store\model
*/
class Goods extends GoodsModel
{
/**
* 获取商品详情
* @param int $goodsId
* @return mixed
* @throws BaseException
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getDetail(int $goodsId)
{
// 获取商品基础信息
$goodsInfo = $this->getBasic($goodsId);
9 months ago
if (isset($goodsInfo['skuList1'])) {
$goodsInfo['skuList'] = $goodsInfo['skuList1'];
unset($goodsInfo['skuList1']);
}
11 months ago
// 分类ID集
$goodsInfo['categoryIds'] = GoodsCategoryRelModel::getCategoryIds($goodsInfo['goods_id']);
// 商品多规格属性列表
if ($goodsInfo['spec_type'] == GoodsSpecTypeEnum::MULTI) {
$goodsInfo['specList'] = GoodsSpecRelModel::getSpecList($goodsInfo['goods_id']);
}
// 服务与承诺
$goodsInfo['serviceIds'] = GoodsServiceRelModel::getServiceIds($goodsInfo['goods_id']);
// 商品规格是否锁定(锁定状态下不允许编辑规格)
$goodsInfo['isSpecLocked'] = GoodsService::checkSpecLocked($goodsId);
// 返回商品详细信息
return $goodsInfo;
}
/**
* 获取商品基础信息
* @param int $goodsId
* @return mixed
* @throws BaseException
*/
public function getBasic(int $goodsId)
{
9 months ago
$info = $this->field('spec_type,goods_id')->find($goodsId);
11 months ago
// 关联查询
9 months ago
if ($info->spec_type == GoodsSpecTypeEnum::SINGLE) {
$with = ['images.file', 'skuList1.image', 'video', 'videoCover'];
} else {
$with = ['images.file', 'skuList.image', 'video', 'videoCover'];
}
11 months ago
// 获取商品记录
$goodsInfo = static::detail($goodsId, $with);
9 months ago
// echo "<pre>";
// print_r($goodsInfo->toArray());
// exit();
11 months ago
empty($goodsInfo) && throwError('很抱歉,商品信息不存在');
// 整理商品数据并返回
return parent::setGoodsData($goodsInfo);
}
/**
* 添加商品
* @param array $data
* @return bool
* @throws \cores\exception\BaseException
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function add(array $data): bool
{
// 创建商品数据
$data = $this->createData($data);
// 事务处理
$this->transaction(function () use ($data) {
9 months ago
$data['data_type'] = $data['data_type'] ?? 1;
11 months ago
// 添加商品
$this->save($data);
// 新增商品与分类关联
GoodsCategoryRelModel::increased((int)$this['goods_id'], $data['categoryIds']);
// 新增商品与图片关联
GoodsImageModel::increased((int)$this['goods_id'], $data['imagesIds']);
// 新增商品与规格关联
GoodsSpecRelModel::increased((int)$this['goods_id'], $data['newSpecList']);
// 新增商品sku信息
GoodsSkuModel::add((int)$this['goods_id'], $data['spec_type'], $data['newSkuList']);
// 新增服务与承诺关联
8 months ago
GoodsServiceRelModel::increased((int)$this['goods_id'], $data['serviceIds'] ?? []);
11 months ago
});
return true;
}
/**
* 编辑商品
* @param array $data
* @return bool
* @throws \cores\exception\BaseException
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
9 months ago
public function edit(array $data, $goodsModel = null): bool
11 months ago
{
9 months ago
11 months ago
// 创建商品数据
$data = $this->createData($data);
// 事务处理
9 months ago
$ret = $this->transaction(function () use ($data, $goodsModel) {
9 months ago
//如果是编辑多规格商品
if ($data['spec_type'] == GoodsSpecTypeEnum::MULTI) {
$data['spu_id'] = $this['goods_id'];
}
11 months ago
// 更新商品
$this->save($data);
// 更新商品与分类关联
GoodsCategoryRelModel::updates((int)$this['goods_id'], $data['categoryIds']);
// 更新商品与图片关联
GoodsImageModel::updates((int)$this['goods_id'], $data['imagesIds']);
// 更新商品与规格关联
GoodsSpecRelModel::updates((int)$this['goods_id'], $data['newSpecList']);
// 更新商品sku信息
9 months ago
$ret = GoodsSkuModel::edit((int)$this['goods_id'], $data['spec_type'], $data['newSkuList']);
if ($ret !== true) {
$this->error = $ret;
return false;
}
11 months ago
// 更新服务与承诺关联
8 months ago
GoodsServiceRelModel::updates((int)$this['goods_id'], $data['serviceIds'] ?? []);
9 months ago
return true;
11 months ago
});
9 months ago
// var_dump($ret);
// exit();
return $ret;
11 months ago
}
9 months ago
/**
* 修改商品状态
* @param array $goodsIds 商品id集
* @param bool $state 为true表示上架
* @return bool|false
*/
9 months ago
public function setIsPool(array $goodsIds, int $is_pool): bool
9 months ago
{
// 批量更新记录
return static::updateBase(['is_pool' => $is_pool], [['goods_id', 'in', $goodsIds]]);
}
/**
* 修改商品状态
* @param array $goodsIds 商品id集
* @param bool $state 为true表示上架
* @return bool|false
*/
9 months ago
public function setIsSale(array $goodsIds, int $is_sale): bool
9 months ago
{
// 批量更新记录
return static::updateBase(['is_sale' => $is_sale], [['goods_id', 'in', $goodsIds]]);
}
11 months ago
/**
* 修改商品状态
* @param array $goodsIds 商品id集
* @param bool $state 为true表示上架
* @return bool|false
*/
public function setStatus(array $goodsIds, bool $state): bool
{
// 批量更新记录
return static::updateBase(['status' => $state ? 10 : 20], [['goods_id', 'in', $goodsIds]]);
}
/**
* 软删除
* @param array $goodsIds
* @return bool
*/
public function setDelete(array $goodsIds): bool
{
foreach ($goodsIds as $goodsId) {
if (!GoodsService::checkIsAllowDelete($goodsId)) {
$this->error = '当前商品正在参与其他活动,不允许删除';
return false;
}
}
// 批量更新记录
return static::updateBase(['is_delete' => 1], [['goods_id', 'in', $goodsIds]]);
}
// 获取已售罄的商品
public function getSoldoutGoodsTotal(): int
{
$filter = [
['stock_total', '=', 0],
['status', '=', GoodsStatusEnum::ON_SALE]
];
return $this->getGoodsTotal($filter);
}
/**
* 获取当前商品总数
* @param array $where
* @return int
*/
public function getGoodsTotal(array $where = []): int
{
return $this->where($where)->where('is_delete', '=', 0)->count();
}
/**
* 创建商品数据
* @param array $data
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @throws \cores\exception\BaseException
*/
private function createData(array $data): array
{
9 months ago
$arr = [];
$sale_areas = (isset($data["sale_areas"]) && $data["sale_areas"]) ? explode("、", $data["sale_areas"]) : [];
if ($sale_areas) {
foreach ($sale_areas as $value) {
$regions = Region::whereIn('name', explode(".", $value))->select()->toArray();
$regions = array_column($regions, null, "level");
$arr[] = $regions[1]['id'].".".$regions[2]['id'];
}
}
8 months ago
$data['goods_sku_no'] = $data['goods_no'] ?? "";
11 months ago
// 默认数据
$data = array_merge($data, [
'line_price' => $data['line_price'] ?? 0,
'content' => $data['content'] ?? '',
'newSpecList' => [],
8 months ago
'data_type' => 1,//数据类型
11 months ago
'newSkuList' => [],
'store_id' => self::$storeId,
9 months ago
'sale_areas' => $arr ? implode("、", $arr) : "",
11 months ago
]);
9 months ago
11 months ago
// 整理商品的价格和库存总量
if ($data['spec_type'] == GoodsSpecTypeEnum::MULTI) {
9 months ago
// $data['stock_total'] = GoodsSkuModel::getStockTotal($data['specData']['skuList']);
// [$data['goods_price_min'], $data['goods_price_max']] = GoodsSkuModel::getGoodsPrices($data['specData']['skuList']);
// [$data['line_price_min'], $data['line_price_max']] = GoodsSkuModel::getLinePrices($data['specData']['skuList']);
11 months ago
} elseif ($data['spec_type'] == GoodsSpecTypeEnum::SINGLE) {
$data['goods_price_min'] = $data['goods_price_max'] = $data['goods_price'];
9 months ago
//$data['line_price_min'] = $data['line_price_max'] = $data['line_price'];
$data['line_price_min'] = $data['line_price_max'] = $data['goods_price'];
11 months ago
$data['stock_total'] = $data['stock_num'];
9 months ago
$data['cost_price_min'] = $data['cost_price']??0;
11 months ago
}
// 规格和sku数据处理
if ($data['spec_type'] == GoodsSpecTypeEnum::MULTI) {
// 验证规格值是否合法
SpecModel::checkSpecData($data['specData']['specList']);
// 生成多规格数据 (携带id)
$data['newSpecList'] = SpecModel::getNewSpecList($data['specData']['specList']);
// 生成skuList (携带goods_sku_id)
$data['newSkuList'] = GoodsSkuModel::getNewSkuList($data['newSpecList'], $data['specData']['skuList']);
} elseif ($data['spec_type'] == GoodsSpecTypeEnum::SINGLE) {
// 生成skuItem
8 months ago
$data['newSkuList'] = helper::pick($data, ['goods_price', 'line_price', 'cost_price','stock_num', 'goods_weight','goods_sku_no']);
11 months ago
}
// 单独设置折扣的配置
$data['is_enable_grade'] == 0 && $data['is_alone_grade'] = 0;
$aloneGradeEquity = [];
if ($data['is_alone_grade'] == 1) {
foreach ($data['alone_grade_equity'] as $key => $value) {
$gradeId = str_replace('grade_id:', '', $key);
$aloneGradeEquity[$gradeId] = $value;
}
}
8 months ago
$profit = (float)$data['goods_price_min'] - (float)$data['cost_price_min'];
$profit_rate = (float)$data['goods_price_min'] > 0 ? bcmul((string)($profit / (float)$data['goods_price_min']) , "100", 2) : 0.00;
$data['profit'] = $profit;
$data['profit_rate'] = $profit_rate;
11 months ago
$data['alone_grade_equity'] = $aloneGradeEquity;
return $data;
}
}