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.
 
 
 
 
 
 
shipin/app/services/activity/coupon/StoreCouponIssueServices.php

667 lines
26 KiB

<?php
// +----------------------------------------------------------------------
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
// +----------------------------------------------------------------------
// | Copyright (c) 2016~2023 https://www.crmeb.com All rights reserved.
// +----------------------------------------------------------------------
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
// +----------------------------------------------------------------------
// | Author: CRMEB Team <admin@crmeb.com>
// +----------------------------------------------------------------------
declare (strict_types=1);
namespace app\services\activity\coupon;
use app\services\BaseServices;
use app\dao\activity\coupon\StoreCouponIssueDao;
use app\services\order\StoreCartServices;
use app\services\product\product\StoreCategoryServices;
use app\services\product\product\StoreProductServices;
use app\services\user\member\MemberCardServices;
use app\services\user\member\MemberRightServices;
use app\services\user\UserServices;
use crmeb\exceptions\AdminException;
use crmeb\exceptions\ApiException;
use crmeb\services\FormBuilder;
use think\facade\Db;
/**
*
* Class StoreCouponIssueServices
* @package app\services\coupon
* @method getUserIssuePrice(string $price) 获取金大于额的优惠卷金额
* @method getCouponInfo($id)
* @method getColumn(array $where, string $field, ?string $key)
* @method productCouponList(array $where, string $field)
* @method checkProductCoupon($product_id)
*/
class StoreCouponIssueServices extends BaseServices
{
public $_couponType = [0 => "通用券", 1 => "品类券", 2 => '商品券'];
/**
* StoreCouponIssueServices constructor.
* @param StoreCouponIssueDao $dao
*/
public function __construct(StoreCouponIssueDao $dao)
{
$this->dao = $dao;
}
/**
* 获取已发布列表
* @param array $where
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getCouponIssueList(array $where)
{
[$page, $limit] = $this->getPageValue();
$where['is_del'] = 0;
$list = $this->dao->getList($where, $page, $limit);
foreach ($list as &$item) {
$item['use_time'] = date('Y-m-d', $item['start_use_time']) . ' ~ ' . date('Y-m-d', $item['end_use_time']);
}
$count = $this->dao->couponCount($where);
return compact('list', 'count');
}
/**
* 获取会员优惠券列表
* @param array $where
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getMemberCouponIssueList(array $where)
{
return $this->dao->getApiIssueList($where);
}
/**
* 新增优惠券
* @param $data
* @return bool
*/
public function saveCoupon($data)
{
if ($data['start_time'] && $data['start_use_time']) {
if ($data['start_time'] < date('Y-m-d 00:00:00')) {
throw new AdminException('开始领取时间不能小于当前时间');
}
if ($data['start_use_time'] < date('Y-m-d 00:00:00')) {
throw new AdminException('开始使用时间不能小于当前时间');
}
if ($data['start_use_time'] < $data['start_time']) {
throw new AdminException(400513);
}
}
if (!in_array((int)$data['receive_type'], [1, 2, 3, 4])) {
throw new AdminException(400758);
}
if (!in_array((int)$data['is_permanent'], [0, 1])) {
throw new AdminException(400758);
}
if (empty($data['coupon_title'])) {
throw new AdminException(400759);
}
if ($data['end_time'] && $data['end_use_time']) {
if ($data['end_use_time'] < $data['end_time']) {
throw new AdminException('用户领取数量不能大于优惠券发布数量');
}
}
$data['start_use_time'] = strtotime((string)$data['start_use_time']);
$data['end_use_time'] = strtotime((string)$data['end_use_time']);
$data['start_time'] = strtotime((string)$data['start_time']);
$data['end_time'] = strtotime((string)$data['end_time']);
$data['title'] = $data['coupon_title'];
$data['remain_count'] = $data['total_count'];
$data['category_id'] = implode(',', $data['category_id']);
if ($data['receive_type'] == 2 || $data['receive_type'] == 3) {
$data['is_permanent'] = 1;
$data['total_count'] = 0;
}
if ($data['is_permanent'] != 1 && $data['receive_limit'] > $data['total_count']) {
throw new AdminException(500031);
}
$data['add_time'] = time();
$res = $this->dao->save($data);
if (($data['product_id'] !== '' || $data['category_id'] !== '') && $res) {
$couponData = [];
if ($data['product_id'] !== '') {
$productIds = explode(',', $data['product_id']);
foreach ($productIds as $product_id) {
$couponData[] = ['product_id' => $product_id, 'coupon_id' => $res->id];
}
} elseif ($data['category_id'] !== '') {
$categoryIds = explode(',', $data['category_id']);
foreach ($categoryIds as $category_id) {
$couponData[] = ['category_id' => $category_id, 'coupon_id' => $res->id];
}
}
/** @var StoreCouponProductServices $storeCouponProductService */
$storeCouponProductService = app()->make(StoreCouponProductServices::class);
$storeCouponProductService->saveAll($couponData);
}
if (!$res) throw new AdminException(100022);
return (int)$res->id;
}
/**
* 修改状态
* @param int $id
* @return array
* @throws \FormBuilder\Exception\FormBuilderException
*/
public function createForm(int $id)
{
$issueInfo = $this->dao->get($id);
if (-1 == $issueInfo['status'] || 1 == $issueInfo['is_del']) throw new AdminException(100007);
$f = [FormBuilder::radio('status', '是否开启', $issueInfo['status'])->options([['label' => '开启', 'value' => 1], ['label' => '关闭', 'value' => 0]])];
return create_form('状态修改', $f, $this->url('/marketing/coupon/released/status/' . $id), 'PUT');
}
/**
* 领取记录
* @param int $id
* @return array
*/
public function issueLog(int $id)
{
$coupon = $this->dao->get($id);
if (!$coupon) {
throw new AdminException(400515);
}
if ($coupon['receive_type'] != 4) {
/** @var StoreCouponIssueUserServices $storeCouponIssueUserService */
$storeCouponIssueUserService = app()->make(StoreCouponIssueUserServices::class);
return $storeCouponIssueUserService->issueLog(['issue_coupon_id' => $id]);
} else {//会员券
/** @var StoreCouponUserServices $storeCouponUserService */
$storeCouponUserService = app()->make(StoreCouponUserServices::class);
return $storeCouponUserService->issueLog(['cid' => $id]);
}
}
/**
* 关注送优惠券
* @param int $uid
* @return bool
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function userFirstSubGiveCoupon(int $uid)
{
$couponList = $this->dao->getGiveCoupon(['receive_type' => 2]);
$this->giveUserCoupon($uid, $couponList ?: []);
return true;
}
/**
* 订单金额达到预设金额赠送优惠卷
* @param $uid
* @param $total_price
* @return bool
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function userTakeOrderGiveCoupon($uid, $total_price)
{
$couponList = $this->dao->getGiveCoupon([['is_full_give', '=', 1], ['full_reduction', '<=', $total_price]]);
$this->giveUserCoupon((int)$uid, $couponList ?: []);
return true;
}
/**
* 下单之后赠送
* @param $uid
* @param $coupon_issue_ids 订单商品关联优惠券ids
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function orderPayGiveCoupon($uid, $coupon_issue_ids)
{
if (!$coupon_issue_ids) return [];
$couponList = $this->dao->getGiveCoupon([['id', 'IN', $coupon_issue_ids]]);
[$couponData, $issueUserData] = $this->giveUserCoupon($uid, $couponList ?: []);
return $couponData;
}
/**
* 发送优惠券
* @param int $uid 发放人id
* @param array $couponList 发送优惠券数据
* @return array[]
*/
public function giveUserCoupon(int $uid, array $couponList)
{
$couponData = $issueUserData = [];
if ($uid && $couponList) {
$time = time();
$ids = array_column($couponList, 'id');
/** @var StoreCouponIssueUserServices $issueUser */
$issueUser = app()->make(StoreCouponIssueUserServices::class);
foreach ($couponList as $item) {
$data['cid'] = $item['id'];
$data['uid'] = $uid;
$data['coupon_title'] = $item['title'];
$data['coupon_price'] = $item['coupon_price'];
$data['use_min_price'] = $item['use_min_price'];
if ($item['coupon_time']) {
$data['add_time'] = $time;
$data['end_time'] = $data['add_time'] + $item['coupon_time'] * 86400;
} else {
$data['add_time'] = $item['start_use_time'];
$data['end_time'] = $item['end_use_time'];
}
$data['type'] = 'send';
$issue['uid'] = $uid;
$issue['issue_coupon_id'] = $item['id'];
$issue['add_time'] = $time;
$issueUserData[] = $issue;
$couponData[] = $data;
unset($data);
unset($issue);
}
if ($couponData) {
/** @var StoreCouponUserServices $storeCouponUser */
$storeCouponUser = app()->make(StoreCouponUserServices::class);
if (!$storeCouponUser->saveAll($couponData)) {
throw new AdminException(100030);
}
}
if ($issueUserData) {
if (!$issueUser->saveAll($issueUserData)) {
throw new AdminException(100031);
}
}
}
return [$couponData, $issueUserData];
}
/**
* 获取优惠券列表
* @param int $uid
* @param array $where
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getIssueCouponList(int $uid, array $where)
{
[$page, $limit] = $this->getPageValue();
$cateId = [];
if ($where['product_id'] == 0) {
if ($where['type'] == -1) { // PC端获取优惠券
$list = $this->dao->getPcIssueCouponList($uid, []);
} else {
$list = $this->dao->getIssueCouponList($uid, (int)$where['type'], 0, $page, $limit);
if (!$list) $list = $this->dao->getIssueCouponList($uid, 1, 0, $page, $limit);
if (!$list) $list = $this->dao->getIssueCouponList($uid, 2, 0, $page, $limit);
}
} else {
/** @var StoreProductServices $storeProductService */
$storeProductService = app()->make(StoreProductServices::class);
/** @var StoreCategoryServices $storeCategoryService */
$storeCategoryService = app()->make(StoreCategoryServices::class);
$cateId = $storeProductService->value(['id' => $where['product_id']], 'cate_id');
$cateId = explode(',', (string)$cateId);
$cateId = array_merge($cateId, $storeCategoryService->cateIdByPid($cateId));
$cateId = array_diff($cateId, [0]);
if ($where['type'] == -1) { // PC端获取优惠券
$list = $this->dao->getPcIssueCouponList($uid, $cateId, $where['product_id']);
} else {
if ($where['type'] == 1) {
$typeId = $cateId;
} elseif ($where['type'] == 2) {
$typeId = $where['product_id'];
} else {
$typeId = 0;
}
$list = $this->dao->getIssueCouponList($uid, (int)$where['type'], $typeId, $page, $limit);
}
}
foreach ($list as &$v) {
$v['coupon_price'] = floatval($v['coupon_price']);
$v['use_min_price'] = floatval($v['use_min_price']);
$v['is_use'] = count($v['used']);
if ($v['end_use_time']) {
$v['start_use_time'] = date('Y/m/d', $v['start_use_time']);
$v['end_use_time'] = date('Y/m/d', $v['end_use_time']);
}
if ($v['start_time']) {
$v['start_time'] = date('Y/m/d', $v['start_time']);
$v['end_time'] = date('Y/m/d', $v['end_time']);
}
}
$data['list'] = $list;
$data['count'] = $this->dao->getIssueCouponCount($where['product_id'], $cateId);
return $data;
}
/**
* 领取优惠券
* @param $id
* @param $user
* @param bool $is_receive
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function issueUserCoupon($id, $user, bool $is_receive = false)
{
$issueCouponInfo = $this->dao->getInfo((int)$id);
if ($user->is_money_level <= 0 && $issueCouponInfo['receive_type'] == 4) {
throw new ApiException(400097);
}
$uid = $user->uid;
if (!$issueCouponInfo) throw new ApiException(400516);
/** @var StoreCouponIssueUserServices $issueUserService */
$issueUserService = app()->make(StoreCouponIssueUserServices::class);
/** @var StoreCouponUserServices $couponUserService */
$couponUserService = app()->make(StoreCouponUserServices::class);
$this->transaction(function () use ($issueUserService, $uid, $id, $couponUserService, $issueCouponInfo, $is_receive) {
$issueUserService->save(['uid' => $uid, 'issue_coupon_id' => $id, 'add_time' => time()]);
$couponUserService->addUserCoupon($uid, $issueCouponInfo, $is_receive ? 'get' : 'send');
if ($issueCouponInfo['total_count'] > 0 && $is_receive) {
$issueCouponInfo['remain_count'] -= 1;
$issueCouponInfo->save();
}
});
}
/**
* 会员发放优惠期券
* @param $id
* @param $uid
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function memberIssueUserCoupon($id, $uid)
{
$issueCouponInfo = $this->dao->getInfo((int)$id);
if ($issueCouponInfo) {
/** @var StoreCouponIssueUserServices $issueUserService */
$issueUserService = app()->make(StoreCouponIssueUserServices::class);
/** @var StoreCouponUserServices $couponUserService */
$couponUserService = app()->make(StoreCouponUserServices::class);
if ($issueCouponInfo->remain_count >= 0 || $issueCouponInfo->is_permanent) {
$this->transaction(function () use ($issueUserService, $uid, $id, $couponUserService, $issueCouponInfo) {
//$issueUserService->save(['uid' => $uid, 'issue_coupon_id' => $id, 'add_time' => time()]);
$couponUserService->addMemberUserCoupon($uid, $issueCouponInfo, "send");
// 如果会员劵需要限制数量时打开
if ($issueCouponInfo['total_count'] > 0) {
$issueCouponInfo['remain_count'] -= 1;
$issueCouponInfo->save();
}
});
}
}
}
/**
* 用户优惠劵列表
* @param int $uid
* @param $types
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getUserCouponList(int $uid, $types)
{
/** @var UserServices $userServices */
$userServices = app()->make(UserServices::class);
if (!$userServices->getUserInfo($uid)) {
throw new ApiException(100100);
}
/** @var StoreCouponUserServices $storeConponUser */
$storeConponUser = app()->make(StoreCouponUserServices::class);
return $storeConponUser->getUserCounpon($uid, $types);
}
/**
* 后台发送优惠券
* @param $coupon
* @param $user
* @return bool
*/
public function setCoupon($coupon, $user)
{
$data = [];
$issueData = [];
/** @var StoreCouponUserServices $storeCouponUser */
$storeCouponUser = app()->make(StoreCouponUserServices::class);
/** @var StoreCouponIssueUserServices $storeCouponIssueUser */
$storeCouponIssueUser = app()->make(StoreCouponIssueUserServices::class);
foreach ($user as $k => $v) {
$data[$k]['cid'] = $coupon['id'];
$data[$k]['uid'] = $v;
$data[$k]['coupon_title'] = $coupon['title'];
$data[$k]['coupon_price'] = $coupon['coupon_price'];
$data[$k]['use_min_price'] = $coupon['use_min_price'];
$data[$k]['add_time'] = time();
if ($coupon['coupon_time']) {
$data[$k]['start_time'] = $data[$k]['add_time'];
$data[$k]['end_time'] = $data[$k]['add_time'] + $coupon['coupon_time'] * 86400;
} else {
$data[$k]['start_time'] = $coupon['start_use_time'];
$data[$k]['end_time'] = $coupon['end_use_time'];
}
$data[$k]['type'] = 'send';
$issueData[$k]['uid'] = $v;
$issueData[$k]['issue_coupon_id'] = $coupon['id'];
$issueData[$k]['add_time'] = time();
}
if (!empty($data)) {
if (!$storeCouponUser->saveAll($data)) {
throw new AdminException(100030);
}
if (!$storeCouponIssueUser->saveAll($issueData)) {
throw new AdminException(100031);
}
return true;
}
}
/**
* 获取下单可使用的优惠券列表
* @param int $uid
* @param $cartId
* @param string $price
* @param bool $new
* @return array
* @throws \Psr\SimpleCache\InvalidArgumentException
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function beUsableCouponList(int $uid, $cartId, bool $new, int $shippingType = 1)
{
/** @var StoreCartServices $services */
$services = app()->make(StoreCartServices::class);
$cartGroup = $services->getUserProductCartListV1($uid, $cartId, $new, [], $shippingType);
/** @var StoreCouponUserServices $coupServices */
$coupServices = app()->make(StoreCouponUserServices::class);
return $coupServices->getUsableCouponList($uid, $cartGroup);
}
/**
* 获取单个优惠券类型
* @param array $where
* @return mixed
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getOne(array $where)
{
if (!$where) throw new AdminException(100100);
return $this->dao->getOne($where);
}
/**
* 俩时间相差月份
* @param $date1
* @param $date2
* @return float|int
*/
public function getMonthNum($date1, $date2)
{
$date1_stamp = strtotime($date1);
$date2_stamp = strtotime($date2);
list($date_1['y'], $date_1['m']) = explode("-", date('Y-m', $date1_stamp));
list($date_2['y'], $date_2['m']) = explode("-", date('Y-m', $date2_stamp));
return abs($date_1['y'] - $date_2['y']) * 12 + $date_2['m'] - $date_1['m'];
}
/**
* 给会员发放优惠券
* @param $uid
* @param int $couponId
* @return bool
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function sendMemberCoupon($uid, $couponId = 0)
{
if (!$uid) return false;
/** @var MemberCardServices $memberCardService */
$memberCardService = app()->make(MemberCardServices::class);
//看付费会员是否开启
$isOpenMember = $memberCardService->isOpenMemberCard();
if (!$isOpenMember) return false;
/** @var UserServices $userService */
$userService = app()->make(UserServices::class);
$userInfo = $userService->getUserInfo((int)$uid);
//看是否会员过期
$checkMember = $userService->offMemberLevel($uid, $userInfo);
if (!$checkMember) return false;
/** @var MemberRightServices $memberRightService */
$memberRightService = app()->make(MemberRightServices::class);
//看是否开启会员送券
$isSendCoupon = $memberRightService->getMemberRightStatus("coupon");
if (!$isSendCoupon) return false;
if ($userInfo && (($userInfo['is_money_level'] > 0) || $userInfo['is_ever_level'] == 1)) {
if ($couponId) {//手动点击领取
$couponWhere['id'] = $couponId;
} else {//主动批量发放
$couponWhere['status'] = 1;
$couponWhere['receive_type'] = 4;
$couponWhere['is_del'] = 0;
}
$couponInfo = $this->getMemberCouponIssueList($couponWhere);
if ($couponInfo) {
/** @var StoreCouponUserServices $couponUserService */
$couponUserService = app()->make(StoreCouponUserServices::class);
$couponIds = array_column($couponInfo, 'id');
$couponUserMonth = $couponUserService->memberCouponUserGroupBymonth(['uid' => $uid, 'couponIds' => $couponIds]);
$getTime = array();
if ($couponUserMonth) {
$getTime = array_column($couponUserMonth, 'num', 'time');
}
// 判断这个月是否领取过,而且领全了
//if (in_array(date('Y-m', time()), $getTime)) return false;
$timeKey = date('Y-m', time());
if (array_key_exists($timeKey, $getTime) && $getTime[$timeKey] == count($couponIds)) return false;
$monthNum = $this->getMonthNum(date('Y-m-d H:i:s', time()), date('Y-m-d H:i:s', $userInfo['overdue_time']));
//判断是否领完所有月份
if (count($getTime) >= $monthNum && (array_key_exists($timeKey, $getTime) && $getTime[$timeKey] == count($couponIds)) && $userInfo['is_ever_level'] != 1 && $monthNum > 0) return false;
//看之前是否手动领取过某一张,领取过就不再领取。
$couponUser = $couponUserService->getUserCounponByMonth(['uid' => $uid, 'cid' => $couponIds], 'id,cid');
if ($couponUser) $couponUser = array_combine(array_column($couponUser, 'cid'), $couponUser);
foreach ($couponInfo as $cv) {
if (!isset($couponUser[$cv['id']])) {
$this->memberIssueUserCoupon($cv['id'], $uid);
}
}
}
}
return true;
}
/**
* 获取今日新增优惠券
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getTodayCoupon($uid)
{
$list = $this->dao->getTodayCoupon($uid);
foreach ($list as $key => &$item) {
$item['start_time'] = $item['start_time'] ? date('Y/m/d', $item['start_time']) : 0;
$item['end_time'] = $item['end_time'] ? date('Y/m/d', $item['end_time']) : 0;
$item['coupon_price'] = floatval($item['coupon_price']);
$item['use_min_price'] = floatval($item['use_min_price']);
if (isset($item['used']) && $item['used']) {
unset($list[$key]);
}
}
return array_merge($list);
}
/**
* 获取新人券
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getNewCoupon()
{
$list = $this->dao->getNewCoupon();
foreach ($list as &$item) {
$item['start_time'] = $item['start_time'] ? date('Y/m/d', $item['start_time']) : 0;
$item['end_time'] = $item['end_time'] ? date('Y/m/d', $item['end_time']) : 0;
$item['coupon_price'] = floatval($item['coupon_price']);
$item['use_min_price'] = floatval($item['use_min_price']);
}
return $list;
}
/**
* 获取列表
* @param array $where
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getCouponList(array $where)
{
[$page, $limit] = $this->getPageValue();
$where['is_del'] = 0;
$field = 'id, coupon_title, type, coupon_price, use_min_price, receive_type, is_permanent, add_time, start_time, end_time, start_use_time, end_use_time, coupon_time, status, total_count, remain_count';
$list = $this->dao->getList($where, $page, $limit, $field);
$count = $this->dao->count($where);
return compact('list', 'count');
}
}