Goods.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | 萤火商城系统 [ 致力于通过产品和服务,帮助商家高效化开拓市场 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2017~2021 https://www.yiovo.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed 这不是一个自由软件,不允许对程序代码以任何形式任何目的的再发行
  8. // +----------------------------------------------------------------------
  9. // | Author: 萤火科技 <admin@yiovo.com>
  10. // +----------------------------------------------------------------------
  11. declare (strict_types=1);
  12. namespace app\store\model;
  13. use app\common\library\helper;
  14. use app\store\model\Spec as SpecModel;
  15. use app\common\model\Goods as GoodsModel;
  16. use app\store\model\GoodsSku as GoodsSkuModel;
  17. use app\store\model\GoodsImage as GoodsImageModel;
  18. use app\store\model\GoodsSpecRel as GoodsSpecRelModel;
  19. use app\store\model\goods\ServiceRel as GoodsServiceRelModel;
  20. use app\store\model\GoodsCategoryRel as GoodsCategoryRelModel;
  21. use app\store\service\Goods as GoodsService;
  22. use app\common\enum\goods\SpecType as SpecTypeEnum;
  23. use app\common\enum\goods\Status as GoodsStatusEnum;
  24. use app\common\exception\BaseException;
  25. use app\common\model\GoodsSku;
  26. use think\facade\Db;
  27. use app\store\model\GoodsPackage as GoodsPackageModel;
  28. /**
  29. * 商品模型
  30. * Class Goods
  31. * @package app\store\model
  32. */
  33. class Goods extends GoodsModel
  34. {
  35. /**
  36. * 获取商品详情
  37. * @param int $goodsId
  38. * @return mixed
  39. * @throws BaseException
  40. * @throws \think\db\exception\DataNotFoundException
  41. * @throws \think\db\exception\DbException
  42. * @throws \think\db\exception\ModelNotFoundException
  43. */
  44. public function getDetail(int $goodsId)
  45. {
  46. // 关联查询
  47. $with = ['images' => ['file'], 'skuList' => ['image']];
  48. // 获取商品记录
  49. $goodsInfo = static::detail($goodsId, $with);
  50. empty($goodsInfo) && throwError('很抱歉,商品信息不存');
  51. // 整理商品数据并返回
  52. $goodsInfo = parent::setGoodsData($goodsInfo);
  53. // 分类ID集
  54. $goodsInfo['categoryIds'] = GoodsCategoryRelModel::getCategoryIds($goodsInfo['goods_id']);
  55. // 商品规格列表
  56. $goodsInfo['specList'] = GoodsSpecRelModel::getSpecList($goodsInfo['goods_id']);
  57. // 服务与承诺
  58. $goodsInfo['serviceIds'] = GoodsServiceRelModel::getServiceIds($goodsInfo['goods_id']);
  59. if($goodsInfo['goods_type']==Goods::PACKAGETYPE){
  60. $dataset = [];
  61. $packageList = GoodsPackageModel::where('goods_id',$goodsId)->select();
  62. $skuList =[];
  63. foreach($packageList as $item){
  64. $filter['goods_id'] = $item['rel_goods_id'];
  65. $filter['goods_sku_id'] = $item['rel_goods_sku_id'];
  66. $sku = GoodsSkuModel::where($filter)->find();
  67. if(!empty($sku)){
  68. $sku = $sku->toArray();
  69. $sku['goods_name'] = GoodsModel::where('goods_id',$item['rel_goods_id'])->value("goods_name");
  70. $skuList[] = $sku;
  71. }
  72. }
  73. $goodsInfo['packageSkuData'] = $skuList;
  74. }
  75. // 商品基本信息
  76. return $goodsInfo;
  77. }
  78. /**
  79. * 添加商品
  80. * @param array $data
  81. * @return bool
  82. * @throws \think\db\exception\DataNotFoundException
  83. * @throws \think\db\exception\DbException
  84. * @throws \think\db\exception\ModelNotFoundException
  85. */
  86. public function add(array $data)
  87. {
  88. //判断商品结构的合法性
  89. if(!$this->validateGoods($data)){
  90. return false;
  91. }
  92. $data['goods_type'] = $data['goods_type']??30;
  93. // 创建商品数据
  94. $data = $this->createData($data);
  95. // 事务处理
  96. $this->transaction(function () use ($data) {
  97. // 添加商品
  98. $this->save($data);
  99. // 新增商品与分类关联
  100. GoodsCategoryRelModel::increased((int)$this['goods_id'], $data['categoryIds']);
  101. // 新增商品与图片关联
  102. GoodsImageModel::increased((int)$this['goods_id'], $data['imagesIds']);
  103. //非套餐商品
  104. // if($data['goods_type']!=Goods::PACKAGETYPE){
  105. // 新增商品与规格关联
  106. GoodsSpecRelModel::increased((int)$this['goods_id'], $data['newSpecList']);
  107. // 新增商品sku信息
  108. GoodsSkuModel::add((int)$this['goods_id'], $data['spec_type'], $data['newSkuList']);
  109. // }
  110. // 新增服务与承诺关联
  111. GoodsServiceRelModel::increased((int)$this['goods_id'], $data['serviceIds']);
  112. //套餐 $data
  113. if($data['goods_type']==Goods::PACKAGETYPE){
  114. $packageList = $data['packageSkuData']??[];//sku商品
  115. GoodsPackageModel::add((int)$this['goods_id'], $data['spec_type'], $packageList);
  116. }
  117. });
  118. return true;
  119. }
  120. /**
  121. * 创建商品数据
  122. * @param array $data
  123. * @return array
  124. * @throws \think\db\exception\DataNotFoundException
  125. * @throws \think\db\exception\DbException
  126. * @throws \think\db\exception\ModelNotFoundException
  127. */
  128. private function createData(array $data)
  129. {
  130. if (!$this->goods_id) {
  131. $data['goods_no'] = self::makeGoodsNo(); // 生成商品编号
  132. }
  133. // 默认数据
  134. $data = array_merge($data, [
  135. 'line_price' => $data['line_price'] ?? 0,
  136. 'content' => $data['content'] ?? '',
  137. 'newSpecList' => [],
  138. 'newSkuList' => [],
  139. 'store_id' => self::$storeId,
  140. ]);
  141. // 库存总量 stock_total
  142. // 商品价格 最低最高
  143. if ($data['spec_type'] == SpecTypeEnum::MULTI) {
  144. $data['stock_total'] = GoodsSkuModel::getStockTotal($data['specData']['skuList']);
  145. $data['alarm_stock_total'] = GoodsSkuModel::getAlarmStockTotal($data['specData']['skuList']);
  146. list($data['goods_price_min'], $data['goods_price_max']) = GoodsSkuModel::getGoodsPrices($data['specData']['skuList']);
  147. list($data['line_price_min'], $data['line_price_max']) = GoodsSkuModel::getLinePrices($data['specData']['skuList']);
  148. } elseif ($data['spec_type'] == SpecTypeEnum::SINGLE) {
  149. $data['goods_price_min'] = $data['goods_price_max'] = $data['goods_price'];
  150. $data['line_price_min'] = $data['line_price_max'] = $data['line_price'];
  151. $data['stock_total'] = $data['stock_num'];
  152. $data['alarm_stock_total'] = $data['alarm_stock_num'] ?? 0;
  153. }
  154. // 规格和sku数据处理
  155. if ($data['spec_type'] == SpecTypeEnum::MULTI) {
  156. // 生成多规格数据(携带id)
  157. $data['newSpecList'] = SpecModel::getNewSpecList($data['specData']['specList']);
  158. // 生成skuList ( 携带goods_sku_id )
  159. $data['newSkuList'] = GoodsSkuModel::getNewSkuList($data['newSpecList'], $data['specData']['skuList'], $data['goods_no'], $this->goods_id);
  160. } elseif ($data['spec_type'] == SpecTypeEnum::SINGLE) {
  161. // 生成skuItem
  162. $data['newSkuList'] = helper::pick($data, ['goods_price', 'line_price', 'stock_num', 'alarm_stock_num', 'goods_weight', 'goods_gross_weight','specs']);
  163. $data['newSkuList']['goods_sku_no'] = $data['goods_sku_no'];
  164. if ($this->goods_id) {
  165. $oldSkuInfo = GoodsSkuModel::where('goods_id', $this->goods_id)->where('goods_sku_id', 0)->find();
  166. $data['newSkuList']['clearing_price'] = $oldSkuInfo['clearing_price'] ?? 0;
  167. $data['newSkuList']['platform_rate'] = $oldSkuInfo['platform_rate'] ?? 0;
  168. }
  169. }
  170. // 单独设置折扣的配置
  171. // $data['is_enable_grade'] == 0 && $data['is_alone_grade'] = 0;
  172. // $aloneGradeEquity = [];
  173. // if ($data['is_alone_grade'] == 1) {
  174. // foreach ($data['alone_grade_equity'] as $key => $value) {
  175. // $gradeId = str_replace('grade_id:', '', $key);
  176. // $aloneGradeEquity[$gradeId] = $value;
  177. // }
  178. // }
  179. // $data['alone_grade_equity'] = $aloneGradeEquity;
  180. $data['province_id'] = $data['cascader'][0] ?? 0;
  181. $data['city_id'] = $data['cascader'][1] ?? 0;
  182. $data['region_id'] = $data['cascader'][2] ?? 0;
  183. return $data;
  184. }
  185. /**
  186. * 验证商品的合法性
  187. * @param $data
  188. * @return bool
  189. * @author: zjwhust
  190. * @Time: 2022/1/10 17:58
  191. */
  192. public function validateGoods($data){
  193. // 规格和sku数据处理
  194. if ($data['spec_type'] == SpecTypeEnum::MULTI) {//多规格
  195. $skuList = [];
  196. foreach ($data['specData']['skuList'] as $key=>&$sku) {
  197. if (!isset($sku['goods_sku_no']) || empty($sku['goods_sku_no'])) {//如果没有设置sku编号
  198. $this->error = '商品SKU编码不能为空';
  199. return false;
  200. }
  201. $skuList[] = $sku['goods_sku_no'];
  202. if(!$this->goods_id){ //新增
  203. if(GoodsSkuModel::where(['goods_sku_no'=>$sku['goods_sku_no']])->count()){
  204. $this->error = '商品SKU编码已存在';
  205. return false;
  206. }
  207. }else{ //编辑
  208. if(GoodsSkuModel::where(['goods_sku_no'=>$sku['goods_sku_no']])->where('goods_id','<>',$this->goods_id)->count()){
  209. $this->error = '商品SKU编码已存在';
  210. return false;
  211. }
  212. }
  213. }
  214. if(count(array_unique($skuList))!=count($data['specData']['skuList'])){
  215. $this->error = '商品SKU编码不能重复';
  216. return false;
  217. }
  218. } elseif ($data['spec_type'] == SpecTypeEnum::SINGLE) {//单规格
  219. if(!isset($data['goods_sku_no']) || empty($data['goods_sku_no'])){//如果没有设置sku编号
  220. $this->error = '商品SKU编码不能为空';
  221. return false;
  222. }
  223. if(!$this->goods_id){ //新增
  224. if(GoodsSkuModel::where(['goods_sku_no'=>$data['goods_sku_no']])->count()){
  225. $this->error = '商品SKU编码已存在';
  226. return false;
  227. }
  228. }else{ //编辑
  229. if(GoodsSkuModel::where(['goods_sku_no'=>$data['goods_sku_no']])->where('goods_id','<>',$this->goods_id)->count()){
  230. $this->error = '商品SKU编码已存在';
  231. return false;
  232. }
  233. }
  234. }
  235. return true;
  236. }
  237. /**
  238. * 编辑商品
  239. * @param array $data
  240. * @return bool
  241. * @throws \think\db\exception\DataNotFoundException
  242. * @throws \think\db\exception\DbException
  243. * @throws \think\db\exception\ModelNotFoundException
  244. */
  245. public function edit(array $data)
  246. {
  247. //判断商品结构的合法性
  248. if(!$this->validateGoods($data)){
  249. return false;
  250. }
  251. // 创建商品数据
  252. $data = $this->createData($data);
  253. //判断商品结构的合法性
  254. $this->validateGoods($data);
  255. // 事务处理
  256. $this->transaction(function () use ($data) {
  257. // 更新商品
  258. $this->save($data);
  259. // 更新商品与分类关联
  260. GoodsCategoryRelModel::updates((int)$this['goods_id'], $data['categoryIds']);
  261. // 更新商品与图片关联
  262. GoodsImageModel::updates((int)$this['goods_id'], $data['imagesIds']);
  263. // 更新商品与规格关联
  264. //非套餐商品
  265. // if($data['goods_type']!=Goods::PACKAGETYPE){
  266. GoodsSpecRelModel::updates((int)$this['goods_id'], $data['newSpecList']);
  267. // 更新商品sku信息
  268. GoodsSkuModel::edit((int)$this['goods_id'], $data['spec_type'], $data['newSkuList']);
  269. // }
  270. // 更新服务与承诺关联
  271. GoodsServiceRelModel::updates((int)$this['goods_id'], $data['serviceIds']);
  272. //套餐商品
  273. // if($data['goods_type']==Goods::PACKAGETYPE){
  274. $packageSkuList = $data['packageSkuData']??[];//sku商品
  275. GoodsPackageModel::edit((int)$this['goods_id'], $data['spec_type'], $packageSkuList);
  276. // }
  277. });
  278. return true;
  279. }
  280. /**
  281. * Notes:生成商品编号
  282. * 规则:SP+年+月+日期+时+分+秒+随机3位数
  283. * Author: zhangs
  284. * DateTime: 2021/10/11 16:11
  285. * @return string
  286. */
  287. public static function makeGoodsNo()
  288. {
  289. return "SP".date('YmdHis') . substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 3);
  290. }
  291. /**
  292. * Notes:生成商品SKU编号
  293. * 规则:商品编码的数字部分+2位数字 ,如2021 09 22 11 56 22 666 01
  294. * Author: zhangs
  295. * DateTime: 2021/10/11 16:15
  296. * @param $goodsNo
  297. * @return mixed
  298. */
  299. public static function makeGoodsSkuNo($goodsNo, $key)
  300. {
  301. return $goodsNo . pad_left($key);
  302. // return $goodsNo . substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 2);
  303. }
  304. /**
  305. * 修改商品状态
  306. * @param array $goodsIds 商品id集
  307. * @param bool $state 为true表示上架
  308. * @return false|int
  309. */
  310. /**
  311. * @param array $goodsIds
  312. * @param bool $state
  313. * @return bool
  314. */
  315. public function setStatus(array $goodsIds, bool $state)
  316. {
  317. // 批量更新记录
  318. return static::updateBase(['status' => $state ? 10 : 20], [['goods_id', 'in', $goodsIds]]);
  319. }
  320. /**
  321. * 软删除
  322. * @param array $goodsIds
  323. * @return bool
  324. */
  325. public function setDelete(array $goodsIds)
  326. {
  327. foreach ($goodsIds as $goodsId) {
  328. if (!GoodsService::checkIsAllowDelete($goodsId)) {
  329. $this->error = '当前商品正在参与其他活动,不允许删除';
  330. return false;
  331. }
  332. }
  333. // 批量更新记录
  334. return static::updateBase(['is_delete' => 1], [['goods_id', 'in', $goodsIds]]);
  335. }
  336. // 获取已售罄的商品
  337. public function getSoldoutGoodsTotal()
  338. {
  339. $filter = [
  340. ['stock_total', '=', 0],
  341. ['status', '=', GoodsStatusEnum::ON_SALE]
  342. ];
  343. return $this->getGoodsTotal($filter);
  344. }
  345. // 获取库存预警的商品
  346. public function getAlarmGoodsTotal()
  347. {
  348. return GoodsSku::alias('sku')
  349. ->leftJoin('goods', 'goods.goods_id=sku.goods_id')
  350. ->where([
  351. ['goods.status', '=', GoodsStatusEnum::ON_SALE],
  352. ['goods.is_delete', '=', 0],
  353. ['sku.stock_num', 'exp', Db::raw('< `alarm_stock_num`')]
  354. ])
  355. ->count();
  356. }
  357. // 获取在售商品总数
  358. public function getOnSaleGoodsTotal()
  359. {
  360. $filter = [
  361. ['status', '=', GoodsStatusEnum::ON_SALE]
  362. ];
  363. return $this->getGoodsTotal($filter);
  364. }
  365. /**
  366. * 获取当前商品总数
  367. * @param array $where
  368. * @return int
  369. */
  370. public function getGoodsTotal($where = [])
  371. {
  372. return $this->where($where)->where('is_delete', '=', 0)->count();
  373. }
  374. }