473 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			473 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
| <?php
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | likeshop开源商城系统
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
 | ||
| // | gitee下载:https://gitee.com/likeshop_gitee
 | ||
| // | github下载:https://github.com/likeshop-github
 | ||
| // | 访问官网:https://www.likeshop.cn
 | ||
| // | 访问社区:https://home.likeshop.cn
 | ||
| // | 访问手册:http://doc.likeshop.cn
 | ||
| // | 微信公众号:likeshop技术社区
 | ||
| // | likeshop系列产品在gitee、github等公开渠道开源版本可免费商用,未经许可不能去除前后端官方版权标识
 | ||
| // |  likeshop系列产品收费版本务必购买商业授权,购买去版权授权后,方可去除前后端官方版权标识
 | ||
| // | 禁止对系统程序代码以任何目的,任何形式的再发布
 | ||
| // | likeshop团队版权所有并拥有最终解释权
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | author: likeshop.cn.team
 | ||
| // +----------------------------------------------------------------------
 | ||
| 
 | ||
| namespace app\shopapi\logic;
 | ||
| 
 | ||
| use app\common\basics\Logic;
 | ||
| use app\common\enum\GoodsEnum;
 | ||
| use app\common\enum\ShopEnum;
 | ||
| use app\common\model\goods\Goods;
 | ||
| use app\common\model\goods\GoodsItem;
 | ||
| use app\common\model\goods\GoodsSpec;
 | ||
| use app\common\server\ConfigServer;
 | ||
| use app\common\server\UrlServer;
 | ||
| use think\facade\Db;
 | ||
| 
 | ||
| class GoodsLogic extends Logic
 | ||
| {
 | ||
|     /**
 | ||
|      * @notes 商品列表
 | ||
|      * @param $params
 | ||
|      * @return array
 | ||
|      * @throws \think\db\exception\DataNotFoundException
 | ||
|      * @throws \think\db\exception\DbException
 | ||
|      * @throws \think\db\exception\ModelNotFoundException
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 11:16
 | ||
|      */
 | ||
|     public function lists($params)
 | ||
|     {
 | ||
|         // 组装条件
 | ||
|         $field = $this->assemblyField();
 | ||
|         $where = $this->assemblyWhere($params);
 | ||
|         $order = $this->assemblyOrder();
 | ||
| 
 | ||
|         $lists = Goods::field($field)
 | ||
|             ->where($where)
 | ||
|             ->page($params['page_no'], $params['page_size'])
 | ||
|             ->order($order)
 | ||
|             ->select()
 | ||
|             ->toArray();
 | ||
| 
 | ||
|         $count = Goods::where($where)->count();
 | ||
| 
 | ||
|         $lists = $this->formatLists($lists);
 | ||
| 
 | ||
|         $more = is_more($count, $params['page_no'], $params['page_size']);
 | ||
|         $btns = $this->btns($params);
 | ||
|         $data = [
 | ||
|             'lists'          => $lists,
 | ||
|             'page_no'       => $params['page_no'],
 | ||
|             'page_size'     => $params['page_size'],
 | ||
|             'count'         => $count,
 | ||
|             'more'          => $more,
 | ||
|             'btns'          => $btns,
 | ||
|         ];
 | ||
| 
 | ||
|         return $data;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 字段
 | ||
|      * @return string[]
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 11:04
 | ||
|      */
 | ||
|     public function assemblyField()
 | ||
|     {
 | ||
|         return [
 | ||
|             "id",
 | ||
|             "name",
 | ||
|             "image",
 | ||
|             "min_price",
 | ||
|             "max_price",
 | ||
|             "stock",
 | ||
|             "sales_actual",
 | ||
|         ];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 搜索条件
 | ||
|      * @param $params
 | ||
|      * @return array[]
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 10:47
 | ||
|      */
 | ||
|     public function assemblyWhere($params)
 | ||
|     {
 | ||
|         // 商家
 | ||
|         $where = [
 | ||
|             ['shop_id', '=', $params['shop_id']]
 | ||
|         ];
 | ||
|         // 商品名称
 | ||
|         if(isset($params['name']) && trim($params['name'])) {
 | ||
|             $where[] = ['name','like','%'.trim($params['name']).'%'];
 | ||
|         }
 | ||
|         // 商品类型 - 默认销售中
 | ||
|         $type = !empty($params['type']) ? (int)$params['type'] : GoodsEnum::SALES;
 | ||
| 
 | ||
|         switch ($type) {
 | ||
|             case GoodsEnum::SALES:
 | ||
|                 $where[] = ['status', '=', GoodsEnum::STATUS_SHELVES];//上架
 | ||
|                 $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
 | ||
|                 $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
 | ||
|                 $where[] = ['stock','exp', Db::raw('>stock_warn')];
 | ||
|                 break;
 | ||
|             case GoodsEnum::WAREHOUSE:
 | ||
|                 $where[] = ['status', '=', GoodsEnum::STATUS_SOLD_OUT];//下架
 | ||
|                 $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
 | ||
|                 $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
 | ||
|                 break;
 | ||
|             case GoodsEnum::WARNING:
 | ||
|                 $where[] = ['status', '=', GoodsEnum::STATUS_SHELVES];//上架
 | ||
|                 $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
 | ||
|                 $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
 | ||
|                 $where[] = ['stock','exp', Db::raw('<=stock_warn')];
 | ||
|                 break;
 | ||
|             case GoodsEnum::RECYCLE_BIN:
 | ||
|                 $where[] = ['del', '=', GoodsEnum::DEL_RECYCLE];
 | ||
|                 $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK];//审核通过
 | ||
|                 break;
 | ||
|             case GoodsEnum::WAIT_AUDIT:
 | ||
|                 $where[] = ['del', '<>', GoodsEnum::DEL_TRUE];
 | ||
|                 $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_STAY];
 | ||
|                 break;
 | ||
|             case GoodsEnum::UNPASS_AUDIT:
 | ||
|                 $where[] = ['del', '<>', GoodsEnum::DEL_TRUE];
 | ||
|                 $where[] = ['audit_status', '=', GoodsEnum::AUDIT_STATUS_REFUSE];
 | ||
|                 break;
 | ||
|             default:
 | ||
|                 $where[] = ['del', '=', GoodsEnum::DEL_NORMAL];
 | ||
|         }
 | ||
| 
 | ||
|         return $where;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 排序
 | ||
|      * @return string[]
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 10:47
 | ||
|      */
 | ||
|     public function assemblyOrder()
 | ||
|     {
 | ||
|         return  [
 | ||
|             'sort' => 'asc',
 | ||
|             'id' => 'desc'
 | ||
|         ];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 格式化
 | ||
|      * @param $lists
 | ||
|      * @return mixed
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 11:06
 | ||
|      */
 | ||
|     public function formatLists($lists)
 | ||
|     {
 | ||
|         if (empty($lists)) {
 | ||
|             return $lists;
 | ||
|         }
 | ||
| 
 | ||
|         foreach ($lists as &$item) {
 | ||
|             $minPrice = floor($item["min_price"] * 100);
 | ||
|             $maxPrice = floor($item["max_price"] * 100);
 | ||
|             $item['price'] = $minPrice == $maxPrice ? "¥" . clearZero($item["min_price"]) : "¥" . clearZero($item["min_price"]) . " ~ " . clearZero($item["max_price"]);
 | ||
|         }
 | ||
| 
 | ||
|         return $lists;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 操作
 | ||
|      * @param $shopId
 | ||
|      * @param $params
 | ||
|      * @return bool
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 14:08
 | ||
|      */
 | ||
|     public function operation($shopId, $params)
 | ||
|     {
 | ||
|         try {
 | ||
|             if (empty($params['action'])) {
 | ||
|                 throw new \Exception("请选择操作");
 | ||
|             }
 | ||
| 
 | ||
|             $goods = Goods::where([
 | ||
|                 'shop_id' => $shopId,
 | ||
|                 'id' => $params['id']
 | ||
|             ])->findOrEmpty();
 | ||
|             if ($goods->isEmpty()) {
 | ||
|                 throw new \Exception("商品不存在");
 | ||
|             }
 | ||
| 
 | ||
|             switch ($params['action']) {
 | ||
|                 case "delete":
 | ||
|                     $this->delete($goods);
 | ||
|                     break;
 | ||
|                 case "recycle":
 | ||
|                     $this->recycle($goods);
 | ||
|                     break;
 | ||
|                 case "on_shelf":
 | ||
|                     $this->onShelf($goods);
 | ||
|                     break;
 | ||
|                 case "off_shelf":
 | ||
|                     $this->offShelf($goods);
 | ||
|                     break;
 | ||
|                 case "warehouse":
 | ||
|                     $this->warehouse($goods);
 | ||
|                     break;
 | ||
|                 default:
 | ||
|                     throw new \Exception("无效的操作");
 | ||
|             }
 | ||
|             return true;
 | ||
|         } catch (\Exception $e) {
 | ||
|             self::$error = $e->getMessage();
 | ||
|             return false;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 删除商品
 | ||
|      * @param $goods
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 11:58
 | ||
|      */
 | ||
|     public function delete($goods)
 | ||
|     {
 | ||
|         $goods->del = GoodsEnum::DEL_TRUE;
 | ||
|         $goods->save();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 放入回收站
 | ||
|      * @param $goods
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 11:58
 | ||
|      */
 | ||
|     public function recycle($goods)
 | ||
|     {
 | ||
|         $goods->del = GoodsEnum::DEL_RECYCLE;
 | ||
|         $goods->save();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 上架
 | ||
|      * @param $goods
 | ||
|      * @throws \Exception
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 12:01
 | ||
|      */
 | ||
|     public function onShelf($goods)
 | ||
|     {
 | ||
|         if ($goods->stock <= 0) {
 | ||
|             throw new \Exception("库存不足不允许上架");
 | ||
|         }
 | ||
|         $goods->status = GoodsEnum::STATUS_SHELVES;
 | ||
|         $goods->save();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 下架
 | ||
|      * @param $goods
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 14:04
 | ||
|      */
 | ||
|     public function offShelf($goods)
 | ||
|     {
 | ||
|         $goods->status = GoodsEnum::STATUS_SOLD_OUT;
 | ||
|         $goods->save();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 放入仓库
 | ||
|      * @param $goods
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 14:07
 | ||
|      */
 | ||
|     public function warehouse($goods)
 | ||
|     {
 | ||
|         $goods->status = GoodsEnum::STATUS_SOLD_OUT;
 | ||
|         $goods->del = GoodsEnum::DEL_NORMAL;
 | ||
|         $goods->save();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 商品详情
 | ||
|      * @param $id
 | ||
|      * @return array
 | ||
|      * @throws \think\db\exception\DataNotFoundException
 | ||
|      * @throws \think\db\exception\DbException
 | ||
|      * @throws \think\db\exception\ModelNotFoundException
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 15:14
 | ||
|      */
 | ||
|     public function detail($id)
 | ||
|     {
 | ||
|         $goodsDetail = Goods::with(['goods_image', 'goods_item', 'shop'])
 | ||
|             ->field('id,type,code,name,spec_type,image,video,remark,content,market_price,min_price,max_price,stock,sales_actual,shop_id')
 | ||
|             ->where('id', $id)
 | ||
|             ->findOrEmpty();
 | ||
| 
 | ||
|         if ($goodsDetail->isEmpty()) {
 | ||
|             return [];
 | ||
|         }
 | ||
| 
 | ||
|         $goodsDetail = $goodsDetail->toArray();
 | ||
| 
 | ||
|         // 轮播图添加域名
 | ||
|         foreach($goodsDetail['goods_image'] as &$item) {
 | ||
|             $item['uri'] = empty($item['uri']) ? '' : UrlServer::getFileUrl($item['uri']);
 | ||
|         }
 | ||
| 
 | ||
|         // 规格项及规格值信息
 | ||
|         $goodsDetail['goods_spec'] = GoodsSpec::with('spec_value')
 | ||
|             ->where('goods_id', $goodsDetail['id'])->select()->toArray();
 | ||
| 
 | ||
|         return $goodsDetail;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 商品编辑
 | ||
|      * @param $shopId
 | ||
|      * @param $params
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 15:24
 | ||
|      */
 | ||
|     public function edit($params)
 | ||
|     {
 | ||
|         try {
 | ||
|             $updateData = $this->checkParams($params);
 | ||
|             (new GoodsItem())->saveAll($updateData['itemData']);
 | ||
|             Goods::update($updateData['goodsData']);
 | ||
|             return true;
 | ||
|         } catch (\Exception $e) {
 | ||
|             self::$error = $e->getMessage();
 | ||
|             return false;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 参数校验
 | ||
|      * @param $params
 | ||
|      * @throws \Exception
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/10 15:48
 | ||
|      */
 | ||
|     public function checkParams($params)
 | ||
|     {
 | ||
|         $goodsStock = 0;
 | ||
|         $itemId = 0;
 | ||
|         $updateData = [];
 | ||
| 
 | ||
|         $max_price = 0;
 | ||
|         $min_price = 0;
 | ||
|         $market_price = 0;
 | ||
| 
 | ||
|         if (!isset($params['items']) || !is_array($params['items'])) {
 | ||
|             throw new \Exception("参数缺失或格式有误");
 | ||
|         }
 | ||
|         foreach($params['items'] as $item) {
 | ||
|             if (!is_array($item)) {
 | ||
|                 throw new \Exception("参数格式错误");
 | ||
|             }
 | ||
|             if (!isset($item['id']) || !isset($item['price']) || !isset($item['stock']) || !isset($item['market_price']) || !isset($item['chengben_price'])) {
 | ||
|                 throw new \Exception("参数缺失");
 | ||
|             }
 | ||
|             if ($item['price'] <= 0 || $item['market_price'] <= 0 || $item['chengben_price'] <= 0 || $item['stock'] <= 0) {
 | ||
|                 throw new \Exception("价格及库存均不能为负数和零");
 | ||
|             }
 | ||
|             if ($item['market_price'] < $item['price']) {
 | ||
|                 throw new \Exception("市场价不能低于售价");
 | ||
|             }
 | ||
|             // 重新赋值作用:避免$item字段过多时修改了不该修改的字段
 | ||
|             $temp['id'] = $item['id'];
 | ||
|             $temp['price'] = $item['price'];
 | ||
|             $temp['market_price'] = $item['market_price'];
 | ||
|             $temp['chengben_price'] = $item['chengben_price'];
 | ||
|             $temp['stock'] = $item['stock'];
 | ||
|             // 规格数据
 | ||
|             $updateData[] = $temp;
 | ||
| 
 | ||
|             // 主表库存
 | ||
|             $goodsStock += $item['stock'];
 | ||
|             $itemId = $item['id'];
 | ||
| 
 | ||
|             // 最高价
 | ||
|             if ($item['price'] > $max_price) {
 | ||
|                 $max_price = $item['price'];
 | ||
|             }
 | ||
| 
 | ||
|             if ($min_price == 0) {
 | ||
|                 $min_price = $item['price'];
 | ||
|                 $market_price = $item['market_price'];
 | ||
|             }
 | ||
| 
 | ||
|             //最低价
 | ||
|             if ($item['price'] < $min_price) {
 | ||
|                 $min_price = $item['price'];
 | ||
|                 $market_price = $item['market_price'];
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         // 商品id
 | ||
|         $goodsId = (new GoodsItem())->where(['id' => $itemId])->value('goods_id');
 | ||
| 
 | ||
|         // 商品更新库存
 | ||
|         $goodsData = [
 | ||
|             'id' => $goodsId,
 | ||
|             'stock' => $goodsStock,
 | ||
|             'max_price' => $max_price,
 | ||
|             'min_price' => $min_price,
 | ||
|             'market_price' => $market_price,
 | ||
|         ];
 | ||
| 
 | ||
|         return ['itemData' => $updateData, 'goodsData' => $goodsData];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * @notes 按钮显示与隐藏
 | ||
|      * @param $params
 | ||
|      * @return array
 | ||
|      * @author Tab
 | ||
|      * @date 2021/11/12 17:59
 | ||
|      */
 | ||
|     public function btns($params)
 | ||
|     {
 | ||
|         $recycleBtn = $editBtn = $offShelfBtn = $onShelfBtn = $deleteBtn = $warehouseBtn = false;
 | ||
|         // 商品类型 - 默认销售中
 | ||
|         $type = !empty($params['type']) ? (int)$params['type'] : GoodsEnum::SALES;
 | ||
| 
 | ||
|         switch ($type) {
 | ||
|             case GoodsEnum::SALES:
 | ||
|             case GoodsEnum::WARNING:
 | ||
|                 $recycleBtn = $editBtn = $offShelfBtn = true;
 | ||
|                 break;
 | ||
|             case GoodsEnum::WAREHOUSE:
 | ||
|                 $recycleBtn = $editBtn = $onShelfBtn = true;
 | ||
|                 break;
 | ||
|             case GoodsEnum::RECYCLE_BIN:
 | ||
|                 $deleteBtn = $warehouseBtn = true;
 | ||
|                 break;
 | ||
|             case GoodsEnum::WAIT_AUDIT:
 | ||
|             case GoodsEnum::UNPASS_AUDIT:
 | ||
|                 $deleteBtn = $editBtn = true;
 | ||
|                 break;
 | ||
|         }
 | ||
|         return [
 | ||
|             "recycle_btn" => $recycleBtn,
 | ||
|             "edit_btn" => $editBtn,
 | ||
|             "off_shelf_btn" => $offShelfBtn,
 | ||
|             "on_shelf_btn" => $onShelfBtn,
 | ||
|             "delete_btn" => $deleteBtn,
 | ||
|             "warehouse_btn" => $warehouseBtn,
 | ||
|         ];
 | ||
|     }
 | ||
| } |