199 lines
6.2 KiB
PHP
199 lines
6.2 KiB
PHP
|
<?php
|
||
|
|
||
|
namespace app\traits\account;
|
||
|
|
||
|
use app\exception\ApiRedirectException;
|
||
|
use app\exception\RepositoryException;
|
||
|
use app\model\Account;
|
||
|
use app\model\AccountCoupon;
|
||
|
use app\model\AccountRecord;
|
||
|
use app\model\Coupon;
|
||
|
use app\model\Express;
|
||
|
use app\repository\AccountRepository;
|
||
|
use think\Collection;
|
||
|
use think\db\exception\DataNotFoundException;
|
||
|
use think\db\exception\DbException;
|
||
|
use think\db\exception\ModelNotFoundException;
|
||
|
use think\Model;
|
||
|
use think\Paginator;
|
||
|
|
||
|
/**
|
||
|
* 优惠券相关
|
||
|
*
|
||
|
* Trait CouponTrait
|
||
|
* @package app\traits\order
|
||
|
*/
|
||
|
trait CouponTrait
|
||
|
{
|
||
|
/**
|
||
|
* 领取优惠券
|
||
|
*
|
||
|
* @param int $accountId
|
||
|
* @param int $couponId
|
||
|
* @return void
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
* @throws RepositoryException
|
||
|
* @throws ApiRedirectException
|
||
|
*/
|
||
|
public function getCoupon(int $accountId, int $couponId)
|
||
|
{
|
||
|
|
||
|
if (!$coupon = Coupon::where('id', $couponId)->lock(true)->find()) {
|
||
|
throw new RepositoryException('优惠券不存在或已下架');
|
||
|
}
|
||
|
|
||
|
if (!empty($coupon['deleted_at'])) {
|
||
|
throw new RepositoryException('优惠券不存在或已下架');
|
||
|
}
|
||
|
|
||
|
if (!$account = Account::findById($accountId)) {
|
||
|
throw new RepositoryException('请先登录');
|
||
|
}
|
||
|
|
||
|
if ($coupon['remain'] <= 0) {
|
||
|
throw new RepositoryException('优惠券已领完');
|
||
|
}
|
||
|
|
||
|
if ($coupon['end_at'] <= date('Y-m-d H:i:s')) {
|
||
|
throw new RepositoryException('优惠券已过期');
|
||
|
}
|
||
|
|
||
|
if (AccountCoupon::where('coupon_id', $couponId)->where('account_id', $accountId)->count() > 0) {
|
||
|
throw new RepositoryException('您已领过此券');
|
||
|
}
|
||
|
|
||
|
if ($account['phone_active'] != Account::COMMON_ON) {
|
||
|
throw new ApiRedirectException('请先完善账号资料');
|
||
|
}
|
||
|
|
||
|
AccountCoupon::create([
|
||
|
'account_id' => $accountId,
|
||
|
'coupon_id' => $couponId,
|
||
|
'status' => AccountCoupon::STATUS_NORMAL,
|
||
|
'is_taste' => (int)($coupon['type'] == Coupon::TYPE_TASTE),
|
||
|
'created_at' => date('Y-m-d H:i:s'),
|
||
|
]);
|
||
|
|
||
|
$coupon->inc('receive')->dec('remain')->update();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 用户优惠券详情
|
||
|
*
|
||
|
* @param int $accountId
|
||
|
* @param int $couponId
|
||
|
*/
|
||
|
public function userCouponInfo(int $accountId, int $couponId)
|
||
|
{
|
||
|
return AccountCoupon::findOne(['account_id' => $accountId, 'coupon_id' => $couponId], [], function ($q) {
|
||
|
return $q->with(["coupon"]);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 获取个人优惠券
|
||
|
*
|
||
|
* @param int $accountId
|
||
|
* @param string $status 默认normal=未使用
|
||
|
* @param int $page
|
||
|
* @param int $size 0 不限 获取全部
|
||
|
* @return AccountCoupon[]|array|Collection
|
||
|
* @throws DataNotFoundException
|
||
|
* @throws DbException
|
||
|
* @throws ModelNotFoundException
|
||
|
*/
|
||
|
public function couponList(string $status = AccountCoupon::STATUS_NORMAL, int $accountId = 0, int $page = 1, int $size = 0)
|
||
|
{
|
||
|
$normalList = AccountCoupon::with(['coupon'])->where('account_id', $accountId)
|
||
|
->where('status', AccountCoupon::STATUS_NORMAL)
|
||
|
->select();
|
||
|
$expiredList = $normalList->where('end_at', '<', date('Y-m-d H:i:s'))->column('id');
|
||
|
if (!empty($expiredList)) {
|
||
|
AccountCoupon::whereIn('id', $expiredList)->update(['status' => AccountCoupon::STATUS_INVALID]);
|
||
|
}
|
||
|
|
||
|
if (in_array($status, AccountCoupon::statusList())) {
|
||
|
// 待查询的优惠券ID列表
|
||
|
$withIds = AccountCoupon::where('account_id', $accountId)
|
||
|
->when(!empty($status), function ($q) use ($status) {
|
||
|
$q->where('status', $status);
|
||
|
})
|
||
|
->when($size > 0, function ($q) use ($page, $size) {
|
||
|
$q->page($page, $size);
|
||
|
})
|
||
|
->order('created_at', 'desc')
|
||
|
->column('coupon_id');
|
||
|
return Coupon::whereIn('id', $withIds)
|
||
|
->field('id,name,cover,type,condition,amount,begin_at,end_at')
|
||
|
->select();
|
||
|
}
|
||
|
|
||
|
if ($status == 'waiting') {
|
||
|
//待领取 可领券的优惠券
|
||
|
//排除所有领过优惠券
|
||
|
$now = date('Y-m-d H:i:s');
|
||
|
$hasIds = AccountCoupon::where('account_id', $accountId)->column('coupon_id');
|
||
|
return Coupon::whereNotIn('id', $hasIds)
|
||
|
->whereNull('deleted_at')
|
||
|
->where('begin_at', '<', $now)
|
||
|
->where('end_at', '>', $now)
|
||
|
->where('remain', '>', 0)
|
||
|
->order('sort', 'desc')
|
||
|
->order('created_at', 'desc')
|
||
|
->field('id,name,cover,type,condition,amount,begin_at,end_at')
|
||
|
->select();
|
||
|
}
|
||
|
|
||
|
return new Collection();
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 检测并处理过期优惠券
|
||
|
*
|
||
|
* @param int $accountId
|
||
|
*/
|
||
|
public function checkExpire(int $accountId)
|
||
|
{
|
||
|
$accountCouponList = AccountCoupon::where('account_id', $accountId)
|
||
|
->where('status', AccountCoupon::STATUS_NORMAL)
|
||
|
->column('coupon_id');
|
||
|
$expiredList = Coupon::whereIn('id', $accountCouponList)
|
||
|
->where('end_at', '<=', date('Y-m-d H:i:s'))
|
||
|
->column('id');
|
||
|
if (!empty($expiredList)) {
|
||
|
AccountCoupon::where('account_id', $accountId)
|
||
|
->whereIn('coupon_id', $expiredList)
|
||
|
->save([
|
||
|
'status' => AccountCoupon::STATUS_INVALID,
|
||
|
]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 创建一个体验券展示二维码
|
||
|
*
|
||
|
* @param int $accountId
|
||
|
* @param int $couponId
|
||
|
*/
|
||
|
public function createExperienceCouponQR($accountId, $couponId)
|
||
|
{
|
||
|
return md5($accountId . AccountCoupon::CODE_SALT . $couponId);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 验证一个体验券展示二维码
|
||
|
*
|
||
|
* @param int $accountId
|
||
|
* @param int $couponId
|
||
|
* @param string $secret
|
||
|
*/
|
||
|
public function checkExperienceCouponQR($accountId, $couponId, $secret)
|
||
|
{
|
||
|
return ($secret == md5($accountId . AccountCoupon::CODE_SALT . $couponId)) ? true : "秘钥核验失败";
|
||
|
}
|
||
|
|
||
|
|
||
|
}
|