截流自动化的商城平台
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.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | likeshop开源商城系统
  4. // +----------------------------------------------------------------------
  5. // | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
  6. // | gitee下载:https://gitee.com/likeshop_gitee
  7. // | github下载:https://github.com/likeshop-github
  8. // | 访问官网:https://www.likeshop.cn
  9. // | 访问社区:https://home.likeshop.cn
  10. // | 访问手册:http://doc.likeshop.cn
  11. // | 微信公众号:likeshop技术社区
  12. // | likeshop系列产品在gitee、github等公开渠道开源版本可免费商用,未经许可不能去除前后端官方版权标识
  13. // | likeshop系列产品收费版本务必购买商业授权,购买去版权授权后,方可去除前后端官方版权标识
  14. // | 禁止对系统程序代码以任何目的,任何形式的再发布
  15. // | likeshop团队版权所有并拥有最终解释权
  16. // +----------------------------------------------------------------------
  17. // | author: likeshop.cn.team
  18. // +----------------------------------------------------------------------
  19. namespace app\shopapi\logic;
  20. use app\common\basics\Logic;
  21. use app\common\enum\GoodsEnum;
  22. use app\common\enum\ShopEnum;
  23. use app\common\model\goods\Goods;
  24. use app\common\model\goods\GoodsItem;
  25. use app\common\model\goods\GoodsSpec;
  26. use app\common\server\ConfigServer;
  27. use app\common\server\UrlServer;
  28. use think\facade\Db;
  29. class GoodsLogic extends Logic
  30. {
  31. /**
  32. * @notes 商品列表
  33. * @param $params
  34. * @return array
  35. * @throws \think\db\exception\DataNotFoundException
  36. * @throws \think\db\exception\DbException
  37. * @throws \think\db\exception\ModelNotFoundException
  38. * @author Tab
  39. * @date 2021/11/10 11:16
  40. */
  41. public function lists($params)
  42. {
  43. // 组装条件
  44. $field = $this->assemblyField();
  45. $where = $this->assemblyWhere($params);
  46. $order = $this->assemblyOrder();
  47. $lists = Goods::field($field)
  48. ->where($where)
  49. ->page($params['page_no'], $params['page_size'])
  50. ->order($order)
  51. ->select()
  52. ->toArray();
  53. $count = Goods::where($where)->count();
  54. $lists = $this->formatLists($lists);
  55. $more = is_more($count, $params['page_no'], $params['page_size']);
  56. $btns = $this->btns($params);
  57. $data = [
  58. 'lists' => $lists,
  59. 'page_no' => $params['page_no'],
  60. 'page_size' => $params['page_size'],
  61. 'count' => $count,
  62. 'more' => $more,
  63. 'btns' => $btns,
  64. ];
  65. return $data;
  66. }
  67. /**
  68. * @notes 字段
  69. * @return string[]
  70. * @author Tab
  71. * @date 2021/11/10 11:04
  72. */
  73. public function assemblyField()
  74. {
  75. return [
  76. "id",
  77. "name",
  78. "image",
  79. "min_price",
  80. "max_price",
  81. "stock",
  82. "sales_actual",
  83. ];
  84. }
  85. /**
  86. * @notes 搜索条件
  87. * @param $params
  88. * @return array[]
  89. * @author Tab
  90. * @date 2021/11/10 10:47
  91. */
  92. public function assemblyWhere($params)
  93. {
  94. // 商家
  95. $where = [
  96. ['shop_id', '=', $params['shop_id']]
  97. ];
  98. // 商品名称
  99. if(isset($params['name']) && trim($params['name'])) {
  100. $where[] = ['name','like','%'.trim($params['name']).'%'];
  101. }
  102. // 商品类型 - 默认销售中
  103. $type = !empty($params['type']) ? (int)$params['type'] : GoodsEnum::SALES;
  104. switch ($type) {
  105. case GoodsEnum::SALES:
  106. $where[] = ['status', '=', GoodsEnum::STATUS_SHELVES];//上架
  107. $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
  108. $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
  109. $where[] = ['stock','exp', Db::raw('>stock_warn')];
  110. break;
  111. case GoodsEnum::WAREHOUSE:
  112. $where[] = ['status', '=', GoodsEnum::STATUS_SOLD_OUT];//下架
  113. $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
  114. $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
  115. break;
  116. case GoodsEnum::WARNING:
  117. $where[] = ['status', '=', GoodsEnum::STATUS_SHELVES];//上架
  118. $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
  119. $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
  120. $where[] = ['stock','exp', Db::raw('<=stock_warn')];
  121. break;
  122. case GoodsEnum::RECYCLE_BIN:
  123. $where[] = ['del', '=', GoodsEnum::DEL_RECYCLE];
  124. $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
  125. break;
  126. case GoodsEnum::WAIT_AUDIT:
  127. $where[] = ['del', '<>', GoodsEnum::DEL_TRUE];
  128. $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_STAY];
  129. break;
  130. case GoodsEnum::UNPASS_AUDIT:
  131. $where[] = ['del', '<>', GoodsEnum::DEL_TRUE];
  132. $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_REFUSE];
  133. break;
  134. default:
  135. $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
  136. }
  137. return $where;
  138. }
  139. /**
  140. * @notes 排序
  141. * @return string[]
  142. * @author Tab
  143. * @date 2021/11/10 10:47
  144. */
  145. public function assemblyOrder()
  146. {
  147. return [
  148. 'sort' => 'asc',
  149. 'id' => 'desc'
  150. ];
  151. }
  152. /**
  153. * @notes 格式化
  154. * @param $lists
  155. * @return mixed
  156. * @author Tab
  157. * @date 2021/11/10 11:06
  158. */
  159. public function formatLists($lists)
  160. {
  161. if (empty($lists)) {
  162. return $lists;
  163. }
  164. foreach ($lists as &$item) {
  165. $minPrice = floor($item["min_price"] * 100);
  166. $maxPrice = floor($item["max_price"] * 100);
  167. $item['price'] = $minPrice == $maxPrice ? "¥" . clearZero($item["min_price"]) : "¥" . clearZero($item["min_price"]) . " ~ " . clearZero($item["max_price"]);
  168. }
  169. return $lists;
  170. }
  171. /**
  172. * @notes 操作
  173. * @param $shopId
  174. * @param $params
  175. * @return bool
  176. * @author Tab
  177. * @date 2021/11/10 14:08
  178. */
  179. public function operation($shopId, $params)
  180. {
  181. try {
  182. if (empty($params['action'])) {
  183. throw new \Exception("请选择操作");
  184. }
  185. $goods = Goods::where([
  186. 'shop_id' => $shopId,
  187. 'id' => $params['id']
  188. ])->findOrEmpty();
  189. if ($goods->isEmpty()) {
  190. throw new \Exception("商品不存在");
  191. }
  192. switch ($params['action']) {
  193. case "delete":
  194. $this->delete($goods);
  195. break;
  196. case "recycle":
  197. $this->recycle($goods);
  198. break;
  199. case "on_shelf":
  200. $this->onShelf($goods);
  201. break;
  202. case "off_shelf":
  203. $this->offShelf($goods);
  204. break;
  205. case "warehouse":
  206. $this->warehouse($goods);
  207. break;
  208. default:
  209. throw new \Exception("无效的操作");
  210. }
  211. return true;
  212. } catch (\Exception $e) {
  213. self::$error = $e->getMessage();
  214. return false;
  215. }
  216. }
  217. /**
  218. * @notes 删除商品
  219. * @param $goods
  220. * @author Tab
  221. * @date 2021/11/10 11:58
  222. */
  223. public function delete($goods)
  224. {
  225. $goods->del = GoodsEnum::DEL_TRUE;
  226. $goods->save();
  227. }
  228. /**
  229. * @notes 放入回收站
  230. * @param $goods
  231. * @author Tab
  232. * @date 2021/11/10 11:58
  233. */
  234. public function recycle($goods)
  235. {
  236. $goods->del = GoodsEnum::DEL_RECYCLE;
  237. $goods->save();
  238. }
  239. /**
  240. * @notes 上架
  241. * @param $goods
  242. * @throws \Exception
  243. * @author Tab
  244. * @date 2021/11/10 12:01
  245. */
  246. public function onShelf($goods)
  247. {
  248. if ($goods->stock <= 0) {
  249. throw new \Exception("库存不足不允许上架");
  250. }
  251. $goods->status = GoodsEnum::STATUS_SHELVES;
  252. $goods->save();
  253. }
  254. /**
  255. * @notes 下架
  256. * @param $goods
  257. * @author Tab
  258. * @date 2021/11/10 14:04
  259. */
  260. public function offShelf($goods)
  261. {
  262. $goods->status = GoodsEnum::STATUS_SOLD_OUT;
  263. $goods->save();
  264. }
  265. /**
  266. * @notes 放入仓库
  267. * @param $goods
  268. * @author Tab
  269. * @date 2021/11/10 14:07
  270. */
  271. public function warehouse($goods)
  272. {
  273. $goods->status = GoodsEnum::STATUS_SOLD_OUT;
  274. $goods->del = GoodsEnum::DEL_NORMAL;
  275. $goods->save();
  276. }
  277. /**
  278. * @notes 商品详情
  279. * @param $id
  280. * @return array
  281. * @throws \think\db\exception\DataNotFoundException
  282. * @throws \think\db\exception\DbException
  283. * @throws \think\db\exception\ModelNotFoundException
  284. * @author Tab
  285. * @date 2021/11/10 15:14
  286. */
  287. public function detail($id)
  288. {
  289. $goodsDetail = Goods::with(['goods_image', 'goods_item', 'shop'])
  290. ->field('id,type,code,name,spec_type,image,video,remark,content,market_price,min_price,max_price,stock,sales_actual,shop_id')
  291. ->where('id', $id)
  292. ->findOrEmpty();
  293. if ($goodsDetail->isEmpty()) {
  294. return [];
  295. }
  296. $goodsDetail = $goodsDetail->toArray();
  297. // 轮播图添加域名
  298. foreach($goodsDetail['goods_image'] as &$item) {
  299. $item['uri'] = empty($item['uri']) ? '' : UrlServer::getFileUrl($item['uri']);
  300. }
  301. // 规格项及规格值信息
  302. $goodsDetail['goods_spec'] = GoodsSpec::with('spec_value')
  303. ->where('goods_id', $goodsDetail['id'])->select()->toArray();
  304. return $goodsDetail;
  305. }
  306. /**
  307. * @notes 商品编辑
  308. * @param $shopId
  309. * @param $params
  310. * @author Tab
  311. * @date 2021/11/10 15:24
  312. */
  313. public function edit($params)
  314. {
  315. try {
  316. $updateData = $this->checkParams($params);
  317. (new GoodsItem())->saveAll($updateData['itemData']);
  318. Goods::update($updateData['goodsData']);
  319. return true;
  320. } catch (\Exception $e) {
  321. self::$error = $e->getMessage();
  322. return false;
  323. }
  324. }
  325. /**
  326. * @notes 参数校验
  327. * @param $params
  328. * @throws \Exception
  329. * @author Tab
  330. * @date 2021/11/10 15:48
  331. */
  332. public function checkParams($params)
  333. {
  334. $goodsStock = 0;
  335. $itemId = 0;
  336. $updateData = [];
  337. $max_price = 0;
  338. $min_price = 0;
  339. $market_price = 0;
  340. if (!isset($params['items']) || !is_array($params['items'])) {
  341. throw new \Exception("参数缺失或格式有误");
  342. }
  343. foreach($params['items'] as $item) {
  344. if (!is_array($item)) {
  345. throw new \Exception("参数格式错误");
  346. }
  347. if (!isset($item['id']) || !isset($item['price']) || !isset($item['stock']) || !isset($item['market_price']) || !isset($item['chengben_price'])) {
  348. throw new \Exception("参数缺失");
  349. }
  350. if ($item['price'] <= 0 || $item['market_price'] <= 0 || $item['chengben_price'] <= 0 || $item['stock'] <= 0) {
  351. throw new \Exception("价格及库存均不能为负数和零");
  352. }
  353. if ($item['market_price'] < $item['price']) {
  354. throw new \Exception("市场价不能低于售价");
  355. }
  356. // 重新赋值作用:避免$item字段过多时修改了不该修改的字段
  357. $temp['id'] = $item['id'];
  358. $temp['price'] = $item['price'];
  359. $temp['market_price'] = $item['market_price'];
  360. $temp['chengben_price'] = $item['chengben_price'];
  361. $temp['stock'] = $item['stock'];
  362. // 规格数据
  363. $updateData[] = $temp;
  364. // 主表库存
  365. $goodsStock += $item['stock'];
  366. $itemId = $item['id'];
  367. // 最高价
  368. if ($item['price'] > $max_price) {
  369. $max_price = $item['price'];
  370. }
  371. if ($min_price == 0) {
  372. $min_price = $item['price'];
  373. $market_price = $item['market_price'];
  374. }
  375. //最低价
  376. if ($item['price'] < $min_price) {
  377. $min_price = $item['price'];
  378. $market_price = $item['market_price'];
  379. }
  380. }
  381. // 商品id
  382. $goodsId = (new GoodsItem())->where(['id' => $itemId])->value('goods_id');
  383. // 商品更新库存
  384. $goodsData = [
  385. 'id' => $goodsId,
  386. 'stock' => $goodsStock,
  387. 'max_price' => $max_price,
  388. 'min_price' => $min_price,
  389. 'market_price' => $market_price,
  390. ];
  391. return ['itemData' => $updateData, 'goodsData' => $goodsData];
  392. }
  393. /**
  394. * @notes 按钮显示与隐藏
  395. * @param $params
  396. * @return array
  397. * @author Tab
  398. * @date 2021/11/12 17:59
  399. */
  400. public function btns($params)
  401. {
  402. $recycleBtn = $editBtn = $offShelfBtn = $onShelfBtn = $deleteBtn = $warehouseBtn = false;
  403. // 商品类型 - 默认销售中
  404. $type = !empty($params['type']) ? (int)$params['type'] : GoodsEnum::SALES;
  405. switch ($type) {
  406. case GoodsEnum::SALES:
  407. case GoodsEnum::WARNING:
  408. $recycleBtn = $editBtn = $offShelfBtn = true;
  409. break;
  410. case GoodsEnum::WAREHOUSE:
  411. $recycleBtn = $editBtn = $onShelfBtn = true;
  412. break;
  413. case GoodsEnum::RECYCLE_BIN:
  414. $deleteBtn = $warehouseBtn = true;
  415. break;
  416. case GoodsEnum::WAIT_AUDIT:
  417. case GoodsEnum::UNPASS_AUDIT:
  418. $deleteBtn = $editBtn = true;
  419. break;
  420. }
  421. return [
  422. "recycle_btn" => $recycleBtn,
  423. "edit_btn" => $editBtn,
  424. "off_shelf_btn" => $offShelfBtn,
  425. "on_shelf_btn" => $onShelfBtn,
  426. "delete_btn" => $deleteBtn,
  427. "warehouse_btn" => $warehouseBtn,
  428. ];
  429. }
  430. }