123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- <?php
- namespace app\common\model;
- use app\common\library\helper;
- use app\common\service\Goods as GoodsService;
- /**
- * 商品模型
- * Class Goods
- * @package app\common\model
- */
- class Goods extends BaseModel
- {
- protected $name = 'goods';
- protected $append = ['goods_sales'];
- /**
- * 计算显示销量 (初始销量 + 实际销量)
- * @param $value
- * @param $data
- * @return mixed
- */
- public function getGoodsSalesAttr($value, $data)
- {
- return $data['sales_initial'] + $data['sales_actual'];
- }
- /**
- * 获取器:单独设置折扣的配置
- * @param $json
- * @return mixed
- */
- public function getAloneGradeEquityAttr($json)
- {
- return json_decode($json, true);
- }
- /**
- * 修改器:单独设置折扣的配置
- * @param $data
- * @return mixed
- */
- public function setAloneGradeEquityAttr($data)
- {
- return json_encode($data);
- }
- /**
- * 关联商品分类表
- * @return \think\model\relation\BelongsTo
- */
- public function category()
- {
- return $this->belongsTo('Category');
- }
- /**
- * 关联商品规格表
- * @return \think\model\relation\HasMany
- */
- public function sku()
- {
- return $this->hasMany('GoodsSku')->order(['goods_sku_id' => 'asc']);
- }
- /**
- * 关联商品规格关系表
- * @return \think\model\relation\BelongsToMany
- */
- public function specRel()
- {
- return $this->belongsToMany('SpecValue', 'GoodsSpecRel')->order(['id' => 'asc']);
- }
- /**
- * 关联商品图片表
- * @return \think\model\relation\HasMany
- */
- public function image()
- {
- return $this->hasMany('GoodsImage')->order(['id' => 'asc']);
- }
- /**
- * 关联运费模板表
- * @return \think\model\relation\BelongsTo
- */
- public function delivery()
- {
- return $this->BelongsTo('Delivery');
- }
- /**
- * 关联订单评价表
- * @return \think\model\relation\HasMany
- */
- public function commentData()
- {
- return $this->hasMany('Comment');
- }
- /**
- * 计费方式
- * @param $value
- * @return mixed
- */
- public function getGoodsStatusAttr($value)
- {
- $status = [10 => '上架', 20 => '下架'];
- return ['text' => $status[$value], 'value' => $value];
- }
- /**
- * 获取商品列表
- * @param $param
- * @return mixed
- * @throws \think\exception\DbException
- */
- public function getList($param)
- {
- // 商品列表获取条件
- $params = array_merge([
- 'status' => 10, // 商品状态
- 'category_id' => 0, // 分类id
- 'search' => '', // 搜索关键词
- 'sortType' => 'all', // 排序类型
- 'sortPrice' => false, // 价格排序 高低
- 'listRows' => 15, // 每页数量
- ], $param);
- // 筛选条件
- $filter = [];
- $params['category_id'] > 0 && $filter['category_id'] = ['IN', Category::getSubCategoryId($params['category_id'])];
- $params['status'] > 0 && $filter['goods_status'] = $params['status'];
- !empty($params['search']) && $filter['goods_name'] = ['like', '%' . trim($params['search']) . '%'];
- // 排序规则
- $sort = [];
- if ($params['sortType'] === 'all') {
- $sort = ['goods_sort', 'goods_id' => 'desc'];
- } elseif ($params['sortType'] === 'sales') {
- $sort = ['goods_sales' => 'desc'];
- } elseif ($params['sortType'] === 'price') {
- $sort = $params['sortPrice'] ? ['goods_max_price' => 'desc'] : ['goods_min_price' => 'asc'];
- }
- // 商品表名称
- $tableName = $this->getTable();
- // 多规格商品 最高价与最低价
- $GoodsSku = new GoodsSku;
- $minPriceSql = $GoodsSku->field(['MIN(goods_price)'])
- ->where('goods_id', 'EXP', "= `$tableName`.`goods_id`")->buildSql();
- $maxPriceSql = $GoodsSku->field(['MAX(goods_price)'])
- ->where('goods_id', 'EXP', "= `$tableName`.`goods_id`")->buildSql();
- // 执行查询
- $list = $this
- ->field(['*', '(sales_initial + sales_actual) as goods_sales',
- "$minPriceSql AS goods_min_price",
- "$maxPriceSql AS goods_max_price"
- ])
- ->with(['category', 'image.file', 'sku'])
- ->where('is_delete', '=', 0)
- ->where($filter)
- ->order($sort)
- ->paginate($params['listRows'], false, [
- 'query' => \request()->request()
- ]);
- // 整理列表数据并返回
- return $this->setGoodsListData($list, true);
- }
- /**
- * 设置商品展示的数据
- * @param $data
- * @param bool $isMultiple
- * @param callable $callback
- * @return mixed
- */
- protected function setGoodsListData($data, $isMultiple = true, callable $callback = null)
- {
- if (!$isMultiple) $dataSource = [&$data]; else $dataSource = &$data;
- // 整理商品列表数据
- foreach ($dataSource as &$goods) {
- // 商品主图
- $goods['goods_image'] = $goods['image'][0]['file_path'];
- // 商品默认规格
- $goods['goods_sku'] = $goods['sku'][0];
- // 回调函数
- is_callable($callback) && call_user_func($callback, $goods);
- }
- return $data;
- }
- /**
- * 根据商品id集获取商品列表
- * @param array $goodsIds
- * @param null $status
- * @return false|\PDOStatement|string|\think\Collection
- * @throws \think\db\exception\DataNotFoundException
- * @throws \think\db\exception\ModelNotFoundException
- * @throws \think\exception\DbException
- */
- public function getListByIds($goodsIds, $status = null)
- {
- // 筛选条件
- $filter = ['goods_id' => ['in', $goodsIds]];
- $status > 0 && $filter['goods_status'] = $status;
- if (!empty($goodsIds)) {
- $this->orderRaw('field(goods_id, ' . implode(',', $goodsIds) . ')');
- }
- // 获取商品列表数据
- // ['category', 'image.file', 'sku', 'spec_rel.spec', 'delivery.rule']
- $data = $this->field(['content'], true)
- ->with(['category', 'image.file', 'sku'])
- ->where($filter)
- ->select();
- // 整理列表数据并返回
- return $this->setGoodsListData($data, true);
- }
- /**
- * 商品多规格信息
- * @param \think\Collection $specRel
- * @param \think\Collection $skuData
- * @return array
- */
- public function getManySpecData($specRel, $skuData)
- {
- // spec_attr
- $specAttrData = [];
- foreach ($specRel as $item) {
- if (!isset($specAttrData[$item['spec_id']])) {
- $specAttrData[$item['spec_id']] = [
- 'group_id' => $item['spec']['spec_id'],
- 'group_name' => $item['spec']['spec_name'],
- 'spec_items' => [],
- ];
- }
- $specAttrData[$item['spec_id']]['spec_items'][] = [
- 'item_id' => $item['spec_value_id'],
- 'spec_value' => $item['spec_value'],
- ];
- }
- // spec_list
- $specListData = [];
- foreach ($skuData as $item) {
- $image = (isset($item['image']) && !empty($item['image'])) ? $item['image'] : ['file_id' => 0, 'file_path' => ''];
- $specListData[] = [
- 'goods_sku_id' => $item['goods_sku_id'],
- 'spec_sku_id' => $item['spec_sku_id'],
- 'rows' => [],
- 'form' => [
- 'image_id' => $image['file_id'],
- 'image_path' => $image['file_path'],
- 'goods_no' => $item['goods_no'],
- 'goods_price' => $item['goods_price'],
- 'goods_weight' => $item['goods_weight'],
- 'line_price' => $item['line_price'],
- 'stock_num' => $item['stock_num'],
- ],
- ];
- }
- return ['spec_attr' => array_values($specAttrData), 'spec_list' => $specListData];
- }
- /**
- * 多规格表格数据
- * @param $goods
- * @return array
- */
- public function getManySpecTable(&$goods)
- {
- $specData = $this->getManySpecData($goods['spec_rel'], $goods['sku']);
- $totalRow = count($specData['spec_list']);
- foreach ($specData['spec_list'] as $i => &$sku) {
- $rowData = [];
- $rowCount = 1;
- foreach ($specData['spec_attr'] as $attr) {
- $skuValues = $attr['spec_items'];
- $rowCount *= count($skuValues);
- $anInterBankNum = ($totalRow / $rowCount);
- $point = (($i / $anInterBankNum) % count($skuValues));
- if (0 === ($i % $anInterBankNum)) {
- $rowData[] = [
- 'rowspan' => $anInterBankNum,
- 'item_id' => $skuValues[$point]['item_id'],
- 'spec_value' => $skuValues[$point]['spec_value']
- ];
- }
- }
- $sku['rows'] = $rowData;
- }
- return $specData;
- }
- /**
- * 获取商品详情
- * @param $goodsId
- * @return static
- */
- public static function detail($goodsId)
- {
- /* @var $model self */
- $model = (new static)->with([
- 'category',
- 'image.file',
- 'sku.image',
- 'spec_rel.spec',
- ])->where('goods_id', '=', $goodsId)
- ->find();
- if (empty($model)) {
- return $model;
- }
- // 整理商品数据并返回
- return $model->setGoodsListData($model, false);
- }
- /**
- * 指定的商品规格信息
- * @param static $goods 商品详情
- * @param int $specSkuId
- * @return array|bool
- */
- public static function getGoodsSku($goods, $specSkuId)
- {
- // 获取指定的sku
- $goodsSku = [];
- foreach ($goods['sku'] as $item) {
- if ($item['spec_sku_id'] == $specSkuId) {
- $goodsSku = $item;
- break;
- }
- }
- if (empty($goodsSku)) {
- return false;
- }
- // 多规格文字内容
- $goodsSku['goods_attr'] = '';
- if ($goods['spec_type'] == 20) {
- $specRelData = helper::arrayColumn2Key($goods['spec_rel'], 'spec_value_id');
- $attrs = explode('_', $goodsSku['spec_sku_id']);
- foreach ($attrs as $specValueId) {
- $goodsSku['goods_attr'] .= $specRelData[$specValueId]['spec']['spec_name'] . ':'
- . $specRelData[$specValueId]['spec_value'] . '; ';
- }
- }
- return $goodsSku;
- }
- }
|