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.
276 lines
8.6 KiB
276 lines
8.6 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\api\model;
|
||
|
|
||
|
use app\api\model\Order as OrderModel;
|
||
|
use app\common\model\Comment as CommentModel;
|
||
|
use app\api\service\User as UserService;
|
||
|
use app\common\library\helper;
|
||
|
use cores\exception\BaseException;
|
||
|
|
||
|
/**
|
||
|
* 商品评价模型
|
||
|
* Class Comment
|
||
|
* @package app\api\model
|
||
|
*/
|
||
|
class Comment extends CommentModel
|
||
|
{
|
||
|
/**
|
||
|
* 隐藏字段
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $hidden = [
|
||
|
'user_id',
|
||
|
'status',
|
||
|
'sort',
|
||
|
'order_id',
|
||
|
'goods_id',
|
||
|
'order_goods_id',
|
||
|
'store_id',
|
||
|
'is_delete',
|
||
|
'update_time'
|
||
|
];
|
||
|
|
||
|
/**
|
||
|
* 获取指定商品评价列表
|
||
|
* @param int $goodsId 商品ID
|
||
|
* @param int|null $scoreType 评分 (10好评 20中评 30差评)
|
||
|
* @return \think\Paginator
|
||
|
* @throws \think\db\exception\DbException
|
||
|
*/
|
||
|
public function getCommentList(int $goodsId, int $scoreType = null): \think\Paginator
|
||
|
{
|
||
|
// 获取评价列表记录
|
||
|
$filter = $this->getFilter($goodsId, $scoreType);
|
||
|
$list = $this->where($filter)
|
||
|
->order(['sort' => 'asc', 'create_time' => 'desc'])
|
||
|
->paginate(15);
|
||
|
// 加载商品评价的关联数据
|
||
|
return static::preload($list, ['user.avatar', 'orderGoods', 'images.file']);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取指定商品评价列表 (限制数量, 不分页)
|
||
|
* @param int $goodsId 商品ID
|
||
|
* @param int $limit 限制的数量
|
||
|
* @return \think\Collection
|
||
|
* @throws \think\db\exception\DataNotFoundException
|
||
|
* @throws \think\db\exception\DbException
|
||
|
* @throws \think\db\exception\ModelNotFoundException
|
||
|
*/
|
||
|
public function listRows(int $goodsId, int $limit = 5): \think\Collection
|
||
|
{
|
||
|
$filter = $this->getFilter($goodsId);
|
||
|
return $this->with(['user.avatar'])
|
||
|
->where($filter)
|
||
|
->order(['sort' => 'asc', $this->getPk()])
|
||
|
->limit($limit)
|
||
|
->select();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取指定商品评价总数量
|
||
|
* @param int $goodsId
|
||
|
* @return int
|
||
|
*/
|
||
|
public function rowsTotal(int $goodsId): int
|
||
|
{
|
||
|
$filter = $this->getFilter($goodsId);
|
||
|
return $this->where($filter)->count();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取查询条件
|
||
|
* @param int $goodsId 商品ID
|
||
|
* @param int|null $scoreType 评分 (10好评 20中评 30差评)
|
||
|
* @return array[]
|
||
|
*/
|
||
|
private function getFilter(int $goodsId, int $scoreType = null): array
|
||
|
{
|
||
|
// 筛选条件
|
||
|
$filter = [
|
||
|
['goods_id', '=', $goodsId],
|
||
|
['status', '=', 1],
|
||
|
['is_delete', '=', 0],
|
||
|
];
|
||
|
// 评分
|
||
|
$scoreType > 0 && $filter[] = ['score', '=', $scoreType];
|
||
|
return $filter;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取指定评分总数
|
||
|
* @param int $goodsId
|
||
|
* @return array|null|\think\Model
|
||
|
* @throws \think\db\exception\DataNotFoundException
|
||
|
* @throws \think\db\exception\DbException
|
||
|
* @throws \think\db\exception\ModelNotFoundException
|
||
|
*/
|
||
|
public function getTotal(int $goodsId)
|
||
|
{
|
||
|
return $this->field([
|
||
|
'count(comment_id) AS `all`',
|
||
|
'count(score = 10 OR NULL) AS `praise`',
|
||
|
'count(score = 20 OR NULL) AS `review`',
|
||
|
'count(score = 30 OR NULL) AS `negative`',
|
||
|
])->where([
|
||
|
'goods_id' => $goodsId,
|
||
|
'is_delete' => 0,
|
||
|
'status' => 1
|
||
|
])->find();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 验证订单是否允许评价
|
||
|
* @param OrderModel $order
|
||
|
* @return boolean
|
||
|
*/
|
||
|
public function checkOrderAllowComment(OrderModel $order): bool
|
||
|
{
|
||
|
// 验证订单是否已完成
|
||
|
if ($order['order_status'] != 30) {
|
||
|
$this->error = '该订单未完成,无法评价';
|
||
|
return false;
|
||
|
}
|
||
|
// 验证订单是否已评价
|
||
|
if ($order['is_comment'] == 1) {
|
||
|
$this->error = '该订单已完成评价';
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 根据已完成订单商品 添加评价
|
||
|
* @param OrderModel $order
|
||
|
* @param $goodsList
|
||
|
* @param array $data
|
||
|
* @return boolean
|
||
|
* @throws BaseException
|
||
|
*/
|
||
|
public function increased(OrderModel $order, $goodsList, array $data): bool
|
||
|
{
|
||
|
// 生成 formData
|
||
|
$formData = $this->formatFormData($data);
|
||
|
// 生成评价数据
|
||
|
$data = $this->createCommentData($order['order_id'], $goodsList, $formData);
|
||
|
if (empty($data)) {
|
||
|
$this->error = '没有输入评价内容';
|
||
|
return false;
|
||
|
}
|
||
|
return $this->transaction(function () use ($order, $goodsList, $formData, $data) {
|
||
|
// 记录评价内容
|
||
|
$result = $this->addAll($data);
|
||
|
// 记录评价图片`
|
||
|
$this->saveAllImages($result, $formData);
|
||
|
// 更新订单评价状态
|
||
|
$isComment = count($goodsList) === count($data);
|
||
|
$this->updateOrderIsComment($order, $isComment, $result);
|
||
|
return true;
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 更新订单评价状态
|
||
|
* @param OrderModel $order
|
||
|
* @param $isComment
|
||
|
* @param $commentList
|
||
|
* @return void
|
||
|
*/
|
||
|
private function updateOrderIsComment(OrderModel $order, $isComment, $commentList): void
|
||
|
{
|
||
|
// 更新订单商品
|
||
|
$orderGoodsData = [];
|
||
|
foreach ($commentList as $comment) {
|
||
|
$orderGoodsData[] = [
|
||
|
'where' => [
|
||
|
'order_goods_id' => $comment['order_goods_id'],
|
||
|
],
|
||
|
'data' => [
|
||
|
'is_comment' => 1
|
||
|
]
|
||
|
];
|
||
|
}
|
||
|
// 更新订单
|
||
|
$isComment && $order->save(['is_comment' => 1]);
|
||
|
(new OrderGoods)->updateAll($orderGoodsData);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 生成评价数据
|
||
|
* @param int $orderId
|
||
|
* @param $goodsList
|
||
|
* @param array $formData
|
||
|
* @return array
|
||
|
* @throws BaseException
|
||
|
*/
|
||
|
private function createCommentData(int $orderId, $goodsList, array $formData): array
|
||
|
{
|
||
|
$data = [];
|
||
|
foreach ($goodsList as $goods) {
|
||
|
if (!isset($formData[$goods['order_goods_id']])) {
|
||
|
throwError('提交的数据不合法');
|
||
|
}
|
||
|
$commentItem = $formData[$goods['order_goods_id']];
|
||
|
$commentItem['content'] = trim($commentItem['content']);
|
||
|
!empty($commentItem['content']) && $data[$goods['order_goods_id']] = [
|
||
|
'score' => $commentItem['score'],
|
||
|
'content' => $commentItem['content'],
|
||
|
'is_picture' => !empty($commentItem['uploaded']),
|
||
|
'sort' => 100,
|
||
|
'status' => 1,
|
||
|
'user_id' => UserService::getCurrentLoginUserId(),
|
||
|
'order_id' => $orderId,
|
||
|
'goods_id' => $commentItem['goods_id'],
|
||
|
'order_goods_id' => $commentItem['order_goods_id'],
|
||
|
'store_id' => self::$storeId
|
||
|
];
|
||
|
}
|
||
|
return $data;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 格式化 formData
|
||
|
* @param array $data
|
||
|
* @return array
|
||
|
*/
|
||
|
private function formatFormData(array $data): array
|
||
|
{
|
||
|
return helper::arrayColumn2Key($data, 'order_goods_id');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 记录评价图片
|
||
|
* @param $commentList
|
||
|
* @param $formData
|
||
|
* @return void
|
||
|
*/
|
||
|
private function saveAllImages($commentList, $formData): void
|
||
|
{
|
||
|
// 生成评价图片数据
|
||
|
$imageData = [];
|
||
|
foreach ($commentList as $comment) {
|
||
|
$item = $formData[$comment['order_goods_id']];
|
||
|
foreach ($item['uploaded'] as $imageId) {
|
||
|
$imageData[] = [
|
||
|
'comment_id' => $comment['comment_id'],
|
||
|
'image_id' => $imageId,
|
||
|
'store_id' => self::$storeId
|
||
|
];
|
||
|
}
|
||
|
}
|
||
|
$model = new CommentImage;
|
||
|
!empty($imageData) && $model->addAll($imageData) !== false;
|
||
|
}
|
||
|
}
|