caipan_shop_admin/app/controller/api/User.php

1220 lines
42 KiB
PHP
Raw Normal View History

2022-05-25 11:35:57 +00:00
<?php
namespace app\controller\api;
use app\exception\ApiRedirectException;
use app\exception\RepositoryException;
use app\exception\TraitException;
use app\model\Account;
use app\model\AccountCoupon;
use app\model\AccountDataLog;
use app\model\AccountRecord;
use app\model\AccountWithdrawalCommission;
use app\model\Message;
use app\repository\AccountRepository;
use app\repository\ArchivesRepository;
use app\repository\OrderRepository;
use app\repository\SpuRepository;
use app\service\File;
use app\service\Jwt;
use app\service\Math;
use app\service\wx\WechatApplets;
use app\validate\User as UserValidate;
use EasyWeChat\Kernel\Exceptions\InvalidConfigException;
use Endroid\QrCode\Builder\Builder;
use Endroid\QrCode\Encoding\Encoding;
use Endroid\QrCode\ErrorCorrectionLevel\ErrorCorrectionLevelHigh;
use Endroid\QrCode\RoundBlockSizeMode\RoundBlockSizeModeMargin;
use Endroid\QrCode\Writer\PngWriter;
use Exception;
use Intervention\Image\Image;
use Intervention\Image\ImageManagerStatic;
use think\db\exception\DataNotFoundException;
use think\db\exception\DbException;
use think\db\exception\ModelNotFoundException;
use think\facade\Config as CConfig;
use think\facade\Db;
use think\facade\Log;
use think\response\Json;
class User extends Base
{
protected $noNeedLogin = [
'login',
'tempLogin',
'feedbackTypes',
'shareUsers',
'homeCoupon',
'getCoupon',
'signByQr',
];
public function index(): Json
{
$userId = $this->request->user['user_id'] ?? 0;
return json(['code' => 0, 'msg' => 'Index page , I am '.$userId]);
}
/**
* 登录 成功返回token及用户信息
*
* @return Json
* @throws InvalidConfigException
*/
public function login(): Json
{
$params = $this->request->param();
Log::error('进入参数');
Log::error($params);
Log::error('参数结束');
$validate = new UserValidate();
if (!$validate->scene('wx_applets')->check($params)) {
return $this->json(4000, $validate->getError());
}
$minApp = WechatApplets::getInstance();
$jsCode = $params['code'];
$wxUser = $minApp->auth->session($jsCode);
if (isset($wxUser['errcode']) && $wxUser['errcode'] != 0) {
return $this->json(4001, $wxUser['errcode'].';'.$wxUser['errmsg'] ?? '登录失败');
}
// $wxUser success has [session_key, openid, unionid]
// 有效期2小时
$wxUser['expire_time'] = time() + 7200;
$wxUser['session_key'] = $wxUser['session_key'] ?? '';
$openID = $wxUser['openid'];
$unionid = $wxUser['unionid'] ?? '';
if (empty($openID) && empty($unionid)) {
return $this->json(4002, '登录失败');
}
$isActive = $params['is_active'] ?? 0;
$isActive = (is_numeric($isActive) && $isActive > 0) ? AccountRepository::BOOL_TRUE : AccountRepository::BOOL_FALSE;
$phoneActive = $params['phone_active'] ?? 0;
$phoneActive = (is_numeric($phoneActive) && $phoneActive > 0) ? AccountRepository::BOOL_TRUE : AccountRepository::BOOL_FALSE;
try {
$repo = AccountRepository::getInstance();
$account = $repo->findByOpenID($openID);
if (!$account && !empty($unionid)) {
$account = $repo->findByUnionId($unionid);
}
$inviteCode = $params['invite_code'] ?? '';
$now = date('Y-m-d H:i:s');
if (!$account) {
// 自动注册
$inviterAid = 0;
$inviterParentAid = 0;
$inviter = null;
// 存在邀请人
if (!empty($inviteCode)) {
$inviter = $repo->findByInviteCode($inviteCode);
if ($inviter) {
$inviterAid = $inviter['id'];
$inviterParentAid = $inviter['inviter_account_id'];
}
}
$account = $repo->create([
'unionid' => $unionid ?? '',
'openid' => $openID,
'last_login' => $now,
'login_ip' => $this->request->ip(),
'created_at' => $now,
'nickname' => $params['nickname'] ?? '',
'headimgurl' => $params['headimgurl'] ?? '',
'country' => $params['country'] ?? '',
'province' => $params['province'] ?? '',
'city' => $params['city'] ?? '',
'county' => $params['county'] ?? '',
'gender' => $params['gender'] ?? 0,
'language' => $params['language'] ?? 'zh_CN',
'mobile' => $params['mobile'] ?? '',
'status' => AccountRepository::STATUS_NORMAL,
'inviter_account_id' => $inviterAid,
'inviter_parent_id' => $inviterParentAid,
'is_active' => $isActive,
'phone_active' => $phoneActive,
'session_key' => $wxUser['session_key'] ?? '',
]);
$regAccountId = $account->id ?? 0;
// 存在邀请人
if ($inviter) {
$repo->addShareRegLog($regAccountId, $inviterAid, $inviterParentAid);
}
AccountRecord::record($regAccountId, AccountRecord::TYPE_OTHER, AccountRecord::ACTION_REGISTER);
} else {
$updateData = [
'last_login' => date('Y-m-d H:i:s'),
'login_ip' => $this->request->ip(),
'session_key' => $wxUser['session_key'] ?? '',
'language' => $params['language'] ?? 'zh_CN',
];
$phoneActiveOld = $account['phone_active'];
// 更新资料
$modifyStringList = ['headimgurl', 'nickname', 'mobile', 'country', 'province', 'city', 'county'];
foreach ($modifyStringList as $modifyKey) {
if (isset($account[$modifyKey]) && empty($account[$modifyKey])) {
$updateData[$modifyKey] = $params[$modifyKey] ?? '';
}
}
if (empty($account['gender'])) {
$updateData['gender'] = $params['gender'] ?? 0;
}
if (isset($account['is_active']) && $account['is_active'] == AccountRepository::BOOL_FALSE) {
$updateData['is_active'] = $isActive;
}
if (isset($account['phone_active']) && $account['phone_active'] == AccountRepository::BOOL_FALSE) {
$updateData['phone_active'] = $phoneActive;
}
$repo->update($updateData, ['id' => $account['id']]);
$account = $repo->findById($account['id']);
// 授权手机号后才能算有效激活用户,并更新相关业务数据
if ($phoneActiveOld == AccountRepository::BOOL_FALSE && $account['phone_active'] == AccountRepository::BOOL_TRUE) {
$repo->activeShareRegLog($account['id']);
}
}
} catch (RepositoryException | Exception $e) {
return $this->json(4003, '登录失败!'.$e->getMessage());
}
$account = $account->toArray();
$jwtData = [
'user_id' => $account['id'],
'open_id' => $openID,
'session_key' => $wxUser['session_key'],
'expire_time' => $wxUser['expire_time'],
];
$account['headimgurl'] = File::convertCompleteFileUrl($account['headimgurl']);
$fields = [
'coding', 'real_name', 'nickname', 'headimgurl', 'gender', 'mobile',
'province', 'city', 'county', 'country', 'birthday',
'score', 'status', 'position', 'invite_code', 'is_active', 'phone_active'
];
$accountData = arrayKeysFilter($account, $fields);
$data = [
'account_id' => $account['id'],
'token' => Jwt::generate($jwtData),
'expire' => $wxUser['expire_time'],
'openid' => $openID,
];
$data = array_merge($data, $accountData);
return $this->json(0, 'success', $data);
}
/**
* 获取用户信息
*
* @return Json
*/
public function info(): Json
{
try {
$accountId = $this->request->user['user_id'] ?? 0;
$fields = [
'id', 'coding', 'real_name', 'nickname', 'headimgurl', 'mobile',
// 'province', 'city', 'county', 'country', 'birthday', 'gender'
'score', 'status', 'invite_code', 'channel',
2022-05-27 09:18:22 +00:00
'is_active', "continuity_sign", "last_sign_online", 'phone_active','address','contacts'
2022-05-25 11:35:57 +00:00
];
$repo = AccountRepository::getInstance();
$user = $repo->findById($accountId, $fields);
if (empty($user)) {
return $this->json(4004, '没有相关的用户记录');
}
$user->headimgurl = File::convertCompleteFileUrl($user->headimgurl ?? '');
/**
* 补充信息:
* 文章收藏量、分享注册用户量包含2级
*/
2022-05-27 09:47:25 +00:00
// $user->collects = $repo->countRecordByAction(AccountRecord::TYPE_CONTENT, AccountRecord::ACTION_COLLECT, $accountId);
// $user->share_users = $repo->shareAccountCount($accountId);
// $user->unread_messages = $repo->countUnReadMessage($accountId);
2022-05-25 11:35:57 +00:00
$user = $user->toArray();
$user = arrayNullToString($user);
$user['order_count'] = OrderRepository::getInstance()->orderCount($accountId);
return $this->json(0, 'success', $user);
} catch (Exception $e) {
return $this->json(4000, '没有相关的用户记录'.$e->getMessage());
}
}
/**
* 首页优惠券
*
* @return Json
* @throws DataNotFoundException
* @throws DbException
* @throws ModelNotFoundException
*/
public function homeCoupon(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
//是否有可领取的推荐优惠券(进入首页时弹出)
$homeCoupon = AccountCoupon::homeCoupon($accountId);
$data['has_coupon'] = (int) !empty($homeCoupon);
$data['home_coupon'] = $homeCoupon;
return $this->json(0, 'success', $data);
}
/**
* 修改用户资料
* 修改内容范围:姓名、昵称、联系电话、性别、地址(省、市、区/县)
* array $params 键值对形式
*/
public function editInfo(): Json
{
if (!$this->request->isPost()) {
return $this->json(4000, '无效请求');
}
$params = $this->request->param();
$validate = new UserValidate();
if (!$validate->scene('edit')->check($params)) {
return $this->json(4001, $validate->getError());
}
$accountId = $this->request->user['user_id'] ?? 0;
try {
AccountRepository::getInstance()->accountEditInfo($accountId, $params);
} catch (RepositoryException $e) {
return $this->json(4002, $e->getMessage());
} catch (Exception $e) {
return $this->json(5000, '资料修改失败');
}
return $this->json();
}
/**
* 临时登录 通过openid登录 仅用于接口测试阶段
*
* @return Json
* @throws DataNotFoundException
* @throws DbException
* @throws ModelNotFoundException
*/
public function tempLogin(): Json
{
$params = $this->request->param();
if (!isset($params['openid'])) {
return $this->json(4001, '参数错误');
}
if (!$user = AccountRepository::getInstance()->findByOpenID($params['openid'])) {
return $this->json(4004, '账号不存在');
}
$data = [
'token' => Jwt::generate(['user_id' => $user['id'], 'nickname' => $user['nickname']]),
'expire' => Jwt::expire()
];
return $this->json(0, 'success', $data);
}
/**
* 个人分享中心
* 统计:累计积分、累计绑定的客户数量(包含2级分销)
*/
public function shareCount(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$res = [
'score' => 0,
'user_count' => [
'first' => 0,
'second' => 0,
'total' => 0,
],
'order_count' => 0,
];
try {
$repo = AccountRepository::getInstance();
$account = $repo->findById($accountId);
if (!empty($account)) {
$res['score'] = $account['score'] ?? 0;
$res['user_count'] = $repo->shareAccountCount($accountId);
}
} catch (Exception $e) {
}
return $this->json(0, 'success', $res);
}
/**
* 查看分享绑定用户的信息
*/
public function shareUsers(): Json
{
$accountId = $this->request->user['user_id'] ?? 2;
$page = $this->request->param('page/d', 1);
$size = $this->request->param('size/d', 10);
$grade = $this->request->param('grade/s', 'first'); // 层级[first|second]
$grade = $grade ?: 'first';
if (!in_array($grade, [AccountRepository::SHARE_GRADE_FIRST, AccountRepository::SHARE_GRADE_SECOND])) {
return $this->json(4001, '层级参数错误');
}
$page = $page <= 0 ? 1 : $page;
$size = $size <= 0 ? 10 : $size;
$list = AccountRepository::getInstance()->shareUsers($accountId, $grade, $page, $size);
return $this->json(0, 'success', $list);
}
/**
* 生成用户个人海报 使用intervention/image库
*/
public function personalPoster(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$bgImg = $this->request->param('bg_src/s', '');
$bgImg = empty($bgImg) ? '' : File::convertCompleteFileUrl($bgImg);
try {
$account = AccountRepository::getInstance()->findById($accountId);
if (empty($account)) {
return $this->json(4000, '无效请求');
}
$domain = $this->request->domain();
$inviteCode = $account['invite_code'] ?: md5($account['id']);
$inviteUrl = $domain.'/share?invite_code='.$inviteCode;
$logoImg = app()->getRootPath().'public/static/images/icon-logo.jpg';
$result = Builder::create()
->writer(new PngWriter())
->writerOptions([])
->data($inviteUrl)
->encoding(new Encoding('UTF-8'))
->errorCorrectionLevel(new ErrorCorrectionLevelHigh())
->size(275)
->margin(10)
->roundBlockSizeMode(new RoundBlockSizeModeMargin())
->logoPath($logoImg)
->logoResizeToHeight(100)
->logoResizeToWidth(100)
->logoPunchoutBackground(true)
->build();
$qrBase64 = $result->getDataUri();
$bgImg = $bgImg ?: app()->getRootPath().'public/static/images/poster-bg1.png';
$headimg = $account['headimgurl'] ?? app()->getRootPath().'public/static/images/icon-logo.jpg';
$name = mb_substr($account['nickname'], 0, 4);
2022-06-06 07:15:12 +00:00
$name = $name ?: '商城';
2022-05-25 11:35:57 +00:00
// 圆形头像
$head = $this->getCircleImg($headimg, 60, 60);
$poster = (string) ImageManagerStatic::make($bgImg)
->insert($qrBase64, 'bottom-right', 10, 100)
->insert($head, 'bottom-left', 30, 80)
->text($name.'...邀您关注', 100, 1080, function ($font) {
$font->file('./static/bb4171.ttf');
$font->size(36);
$font->color('#000000');
$font->align('left');
$font->valign('top');
})
->encode('data-url');
// ->save('aaa.png');
return $this->json(0, 'success', ['poster' => $poster]);
} catch (Exception $e) {
AccountRepository::log('个人海报生成失败', $e);
return $this->json(5000, '个人海报生成失败');
}
}
/**
* 圆形头像.
*
* @see https://github.com/Intervention/image/issues/223
*
* @param $img
* @param int $width
* @param int $height
* @return Image
*/
protected function getCircleImg($img, int $width, int $height): Image
{
$avatar_img = ImageManagerStatic::make($img)->resize($width, $height);
$r = $width / 2;
$temp_image = ImageManagerStatic::canvas(185, 185);
for ($x = 0; $x < $avatar_img->width(); $x++) {
for ($y = 0; $y < $avatar_img->height(); $y++) {
if (((($x - $r) * ($x - $r) + ($y - $r) * ($y - $r)) < ($r * $r))) {
$c = $avatar_img->pickColor($x, $y, 'array');
$temp_image->pixel($c, $x, $y);
}
}
}
return $temp_image;
}
/**
* 用户操作记录 分享、咨询
*
* @return Json
* @throws Exception
*/
public function record(): Json
{
if (!$this->request->isPost()) {
return $this->json(4000, '无效请求');
}
$accountId = $this->request->user['user_id'] ?? 0;
$params = input('post.');
$rules = [
'type|操作类型' => 'require|in:content,other,spu,activity',
'action|操作' => 'require',
];
if (isset($params['type']) && $params['type'] == AccountRecord::TYPE_CONTENT) {
$rules['id|ID'] = 'require';
}
$validate = $this->validateByApi($params, $rules, ['type.in' => '类型错误', 'id.require' => '此类型 ID必传']);
if ($validate !== true) {
return $validate;
}
try {
$relationId = $params['id'] ?? 0;
$relationId = is_numeric($relationId) ? $relationId : 0;
AccountRecord::record($accountId, $params['type'], $params['action'], $relationId);
} catch (Exception $e) {
AccountRepository::log('记录用户操作失败', $e);
return $this->json(5001, '操作失败');
}
return $this->json();
}
/**
* 个人二维码
*/
public function personalQr(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
try {
$account = AccountRepository::getInstance()->findById($accountId);
if (empty($account)) {
return $this->json(4000, '无效请求');
}
$nonceStr = uniqid();
$qrDataStr = json_encode([
'user_coding' => $account['coding'],
'nonce_str' => $nonceStr,
]);
$logoImg = File::convertCompleteFileUrl($account['headimgurl']);
if (empty($logoImg)) {
$logoImg = app()->getRootPath().'public/static/images/icon-logo.jpg';
}
$result = Builder::create()
->writer(new PngWriter())
->writerOptions([])
->data($qrDataStr)
->encoding(new Encoding('UTF-8'))
->errorCorrectionLevel(new ErrorCorrectionLevelHigh())
->size(300)
->margin(10)
->roundBlockSizeMode(new RoundBlockSizeModeMargin())
->logoPath($logoImg)
->logoResizeToHeight(100)
->logoResizeToWidth(100)
->logoPunchoutBackground(true)
->build();
$dataUri = $result->getDataUri();
return $this->json(0, 'success', ['qr' => $dataUri, 'nonce_str' => $nonceStr]);
} catch (Exception $e) {
AccountRepository::log('个人二维码生成失败', $e);
return $this->json(5000, '个人二维码生成失败');
}
}
/**
* 提交反馈意见
*/
public function addFeedback(): Json
{
if (!$this->request->isPost()) {
return $this->json(4000, '无效请求');
}
$accountId = $this->request->user['user_id'] ?? 0;
$params = [
'type_id' => $this->request->param('type_id/d', 0),
'content' => postFilter($this->request->param('content/s', '')),
'user_name' => postFilter($this->request->param('user_name/s', '')),
'user_phone' => postFilter($this->request->param('user_phone/s', '')),
];
try {
AccountRepository::getInstance()->addFeedback($accountId, $params);
} catch (TraitException $e) {
return $this->json(4001, $e->getMessage());
}
return $this->json(0, '您已提交成功,感谢您的反馈!');
}
/**
* 用户消息
*/
public function messages(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$page = $this->request->param('page/d', 1);
$size = $this->request->param('size/d', 10);
$type = $this->request->param('type/s', '');
$page = $page <= 0 ? 1 : $page;
$size = $size <= 0 ? 10 : $size;
$repo = AccountRepository::getInstance();
$whereMap = [];
$fields = [
'id', 'type', 'target', 'title', 'summary', 'content', 'status',
'sort', 'created_at', 'send_at'
];
$orders = ['sort' => 'desc', 'id' => 'desc'];
$regTime = '';
$account = $repo->findById($accountId);
if ($account) {
$regTime = $account['created_at'] ?? '';
}
if ($type) {
if ($type == 'message') {
$whereMap[] = ['type', 'in', [Message::TYPE_NOTICE, Message::TYPE_SYSTEM]];
} else {
$whereMap[] = ['type', '=', $type];
}
}
$resp = $repo->messageList($whereMap, $fields, $page, $size, function ($q) use ($accountId, $regTime) {
return $q->whereTime('send_at', '<=', date('Y-m-d H:i:s'))
->when(!empty($regTime), function ($q3) use ($regTime) {
$q3->whereTime('send_at', '>=', $regTime);
})
->where(function ($q3) use ($accountId) {
$q3->whereRaw('(target = "part" and find_in_set("'.$accountId.'", target_list)) or target <> "part"');
});
}, $orders);
$msgIds = $resp['list']->column('id');
$hadReadMsgIds = $repo->getHadReadMessageIds($accountId, $msgIds);
$msgTypeTextList = $repo->messageTypeTextList();
foreach ($resp['list'] as $item) {
$item['type_text'] = $msgTypeTextList[$item['type']] ?? '';
$item['is_read'] = in_array($item['id'], $hadReadMsgIds) ? AccountRepository::BOOL_TRUE : AccountRepository::BOOL_FALSE;
}
// todo 获取列表后更新阅览日志记录
$repo->addReadLogs($accountId, $msgIds);
return $this->json(0, 'success', $resp);
}
/**
* 查看消息详情
* 添加阅览日志记录
*/
public function messageInfo(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$msgId = $this->request->param('message_id/d', 0);
$repo = AccountRepository::getInstance();
$fields = [
'id', 'type', 'target', 'title', 'summary', 'content', 'target_list',
'status', 'sort', 'created_at', 'send_at'
];
$message = $repo->messageInfo($msgId, $fields);
if (empty($message)) {
return $this->json(4004, '没有相关的消息记录');
}
if ($message['target'] == Message::TARGET_PART) {
$targetList = empty($message['target_list']) ? [] : explode(',', $message['target_list']);
if (!in_array($accountId, $targetList)) {
return $this->json(4001, '没有查看权限');
}
} elseif (!empty($message['send_at']) && strtotime($message['send_at']) > time()) {
return $this->json(4001, '没有查看权限');
}
$repo->addReadLogs($accountId, [$msgId]);
$msgTypeTextList = $repo->messageTypeTextList();
$message['type_text'] = $msgTypeTextList[$message['type']] ?? '';
$message['is_read'] = AccountRepository::BOOL_TRUE;
unset($message['target_list']);
return $this->json(0, 'success', $message);
}
/**
* 客户列表
*
* @return Json
* @throws Exception
*/
public function customer(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$type = input('type/s', 'all');
$keyword = input('keyword/s', '');
$page = input('page/d', 1);
$size = input('size/d', 20);
try {
if ($accountId == 0) {
return $this->json(6001, '请先登录');
}
$where = [];
if ($type !== 'all') {
$where[] = ['channel', '=', $type];
}
if (!empty($keyword)) {
$where[] = ['nickname|real_name|mobile', 'like', '%'.$keyword.'%'];
}
$where[] = ['phone_active', '=', Account::COMMON_ON];
$field = ['id', 'nickname', 'real_name', 'headimgurl', 'mobile', 'channel', 'created_at', 'customer_service', 'coin_total'];
$data = AccountRepository::getInstance()->getAndHandleAccountList($where, $field, $page, $size);
$data['list'] = $data['list']->each(function ($item) {
$item->real_name = $item->real_name ?? '';
unset($item->tags);
});
return $this->json(0, 'success', $data);
} catch (RepositoryException $e) {
return $this->json(4001, $e->getMessage());
} catch (Exception $e) {
ArchivesRepository::log($e->getMessage(), $e);
return $this->json(5000, '操作失败');
}
}
/**
* 绑定手机
*
* @return bool|Json
* @throws Exception
*/
public function bindPhone()
{
if (!$this->request->isPost()) {
return $this->json(4000, '无效请求');
}
$accountId = $this->request->user['user_id'] ?? 0;
$params = input('post.');
$rules = [
'encryptedData|加密数据' => 'require',
'iv|IV' => 'require',
];
$validate = $this->validateByApi($params, $rules);
if ($validate !== true) {
return $validate;
}
try {
if (!$account = Account::findById($accountId)) {
return $this->json(4000, '用户不存在');
}
$oldPhoneActive = $account['phone_active'];
// 解密手机相关数据 若存在手机则覆盖
$minApp = WechatApplets::getInstance();
$sessionKey = $this->request->user['session_key'] ?? '';
$decryptData = $minApp->encryptor->decryptData($sessionKey, $params['iv'], $params['encryptedData']);
$phone = $decryptData['phoneNumber'] ?? ''; // 通过iv和加密数据 解密出手机号
if (Account::where('id', '<>', $accountId)->where('mobile', $phone)->count() > 0) {
return $this->json(4000, '该手机已被绑定,若有绑定错误,请联系客服');
}
if ($phone) {
$account->save(['mobile' => $phone, 'phone_active' => Account::COMMON_ON]);
}
if ($oldPhoneActive == Account::COMMON_OFF) {
AccountRepository::getInstance()->activeShareRegLog($account['id']);
}
return $this->json(0, 'success', ['phone' => $phone]);
} catch (Exception $e) {
AccountRepository::log('手机绑定失败', $e);
return $this->json(5001, '手机绑定失败');
}
}
/**
* 领取优惠券
*
* @return Json
* @throws Exception
*/
public function getCoupon(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$couponID = input('coupon_id/d', 0);
try {
AccountRepository::getInstance()->getCoupon($accountId, $couponID);
return $this->json();
} catch (ApiRedirectException $e) {
return $this->json(302, $e->getMessage());
} catch (RepositoryException $e) {
return $this->json(4000, $e->getMessage());
} catch (Exception $e) {
AccountRepository::log('领取优惠券失败', $e);
return $this->json(5001, '领取优惠券失败');
}
}
/**
* 展示体验券 生成token
*
* @return Json
* @throws Exception
*/
public function exhibitionExperienceCoupon(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$couponID = input('coupon_id/d', 0);
try {
$coupon = AccountRepository::getInstance()->userCouponInfo($accountId, $couponID);
$time = time();
//不存在
if (empty($coupon) || !isset($coupon['coupon'])) {
return $this->json(4002, "优惠券不存在");
}
//状态异常
if ($coupon['status'] != AccountCoupon::STATUS_NORMAL) {
return $this->json(4002, "优惠券状态已经不能使用");
}
//未开始
if (strtotime($coupon['coupon']["begin_at"]) > $time) {
return $this->json(4002, "优惠券还未到开始使用时间");
}
//过期
if (strtotime($coupon['coupon']["end_at"]) < $time) {
return $this->json(4002, "优惠券已过期");
}
if ($coupon['is_taste'] != AccountCoupon::COMMON_ON) {
return $this->json(4002, "优惠券不是体验券");
}
$secret = AccountRepository::getInstance()->createExperienceCouponQR($accountId, $couponID);
$url = ($this->request->domain()."/api/staff/coupon/write-off-experience-coupon?account_id=$accountId&&coupon_id=$couponID&&secret=$secret");
return $this->json(0, "success", ["url" => $url, "secret" => $secret]);
} catch (ApiRedirectException $e) {
return $this->json(302, $e->getMessage());
} catch (RepositoryException $e) {
return $this->json(4000, $e->getMessage());
} catch (Exception $e) {
AccountRepository::log('获取优惠券失败', $e);
return $this->json(5001, '获取优惠券失败');
}
}
/**
* 获取优惠券列表
*
* @return Json
* @throws Exception
*/
public function getCouponList(): Json
{
$accountId = $this->request->user['user_id'] ?? 0;
$status = input('status/s', AccountCoupon::STATUS_NORMAL);
$page = input('page/d', 1);
$size = input('size/d', 0);
try {
$data = AccountRepository::getInstance()->couponList($status, $accountId, $page, $size);
return $this->json(0, 'success', $data);
} catch (Exception $e) {
AccountRepository::log('获取优惠券列表失败', $e);
return $this->json(5001, '获取优惠券列表失败');
}
}
/**
* 获取地址列表
*
* @return Json
* @throws DataNotFoundException
* @throws DbException
* @throws ModelNotFoundException
*/
public function address(): Json
{
$userId = $this->request->user['user_id'] ?? 0;
$list = AccountRepository::getInstance()->findAddressByUid($userId);
return $this->json(0, 'success', $list);
}
/**
* 保存地址
*
* @return Json
*/
public function addressSave(): Json
{
$userId = $this->request->user['user_id'] ?? 0;
$params = $this->request->param();
$validate = new UserValidate();
$params['user_id'] = $userId;
if (!$validate->scene('address')->check($params)) {
return $this->json(4000, $validate->getError());
}
AccountRepository::getInstance()->saveAddress($params);
return $this->json();
}
/**
* 地址详情
*
* @return Json
* @throws DataNotFoundException
* @throws DbException
* @throws ModelNotFoundException
*/
public function addressInfo(): Json
{
if (!$id = input('id/d', 0)) {
return $this->json(4000, '参数错误');
}
$info = AccountRepository::getInstance()->findAddressById($id);
return $this->json(0, 'success', $info);
}
/**
* 地址删除
*
* @return Json
*/
public function addressDel(): Json
{
if (!$id = input('id/d', 0)) {
return $this->json(4000, '参数错误');
}
AccountRepository::getInstance()->delAddress($id);
return $this->json();
}
/**
* 积分收支记录
*
* @return Json
* @throws Exception
*/
public function scoreLog(): Json
{
$page = input('page/d', 1);
$limit = input('size/d', 10);
$type = input("type/s", "in"); //in=收入 out=支出 默认in
$accountId = $this->request->user['user_id'] ?? 0;
$where = [];
$where[] = ["account_id", "=", $accountId];
$where[] = ["type", "=", AccountDataLog::TYPE_SCORE];
if ($type == 'in') {
$where[] = ["num", ">", 0];
}
if ($type == 'out') {
$where[] = ["num", "<", 0];
}
$items = AccountDataLog::findList($where, [], $page, $limit, null, ['id' => 'desc']);
return $this->json(0, '操作成功', $items);
}
/**
* 我的订单
*
* @return Json
* @throws RepositoryException
*/
public function order(): Json
{
$userId = $this->request->user['user_id'] ?? 0;
$page = $this->request->param('page/d', 1);
$size = $this->request->param('size/d', 20);
$tag = $this->request->param('tag/s', 'all');
$domain = $this->request->domain();
$res = OrderRepository::getInstance()->mine($userId, $tag, $page, $size);
2022-06-02 02:07:28 +00:00
$orderStatusTextList = OrderRepository::getInstance()->orderStatusTextList();
$res['list'] = $res['list']->each(function ($item) use ($domain,$orderStatusTextList) {
$item->skus->each(function ($sku) use ($domain,$orderStatusTextList) {
2022-05-25 11:35:57 +00:00
if (!$sku->spec_text) {
$sku->spec_text = [];
$sku->spec_info = [];
} else {
$specArr = json_decode($sku->spec_text, true);
$sku->spec_text = $specArr;
$sku->spec_info = SpuRepository::getInstance()->convertSpecInfo($specArr);
}
2022-06-02 02:07:28 +00:00
$sku->spu_cover = resourceJoin($sku->spu_cover, $domain);
$sku->sku_cover = resourceJoin($sku->sku_cover, $domain);
2022-05-25 11:35:57 +00:00
});
2022-06-02 02:07:28 +00:00
$item->status_text = $orderStatusTextList[$item["status"]];
2022-05-25 11:35:57 +00:00
});
return $this->json(0, 'success', $res);
}
/**
* 立即免拼 需要单人开团允许
*
* @return Json
*/
public function openOne(): Json
{
$userId = $this->request->user['user_id'] ?? 0;
$coding = input('order_coding/s');
try {
if (empty($coding)) {
return $this->json(4001, '参数错误');
}
OrderRepository::getInstance()->openOne($userId, $coding);
return $this->json();
} catch (RepositoryException $e) {
return $this->json(4000, $e->getMessage());
} catch (Exception $e) {
AccountRepository::log('立即免拼失败', $e);
return $this->json(5001, '立即免拼失败');
}
}
/**
* 订单详情
*
* @return Json
* @throws RepositoryException
*/
public function orderDetail(): Json
{
$id = input('id/d', 0);
$userId = $this->request->user['user_id'] ?? 0;
$domain = $this->request->domain();
$res = OrderRepository::getInstance()->detail($id);
2022-06-01 08:18:39 +00:00
$res["status_text"] = OrderRepository::getInstance()->orderStatusTextList()[$res["status"]];
2022-05-25 11:35:57 +00:00
$res->skus->each(function ($sku) use ($domain) {
$specArr = json_decode($sku->spec_text, true);
$sku->spec_text = $specArr;
$sku->spec_info = SpuRepository::getInstance()->convertSpecInfo($specArr);
if (!empty($sku->spu_cover) && !isHttpUrl($sku->spu_cover)) {
$sku->spu_cover = $domain.$sku->spu_cover;
}
if (!empty($sku->sku_cover) && !isHttpUrl($sku->sku_cover)) {
$sku->sku_cover = $domain.$sku->sku_cover;
}
});
if (!$res) {
return $this->json(4000, '订单不存在');
}
if ($res['account_id'] !== $userId) {
return $this->json(4000, '不是您的订单');
}
return $this->json(0, 'success', $res);
}
/**
* 发起提现【佣金】
*
* @return Json
* @throws DataNotFoundException
* @throws DbException
* @throws ModelNotFoundException
*/
public function commissionWithdrawal(): Json
{
$commission = input("commission/f", 0);//单位元
$wechat = input("wechat/s");//微信号
$accountId = $this->request->user['user_id'];
$account = Account::findById($accountId, [], function ($q) {
return $q->lock(true);
});
if (empty($commission) || empty($wechat)) {
return $this->json(4001, '参数错误');
}
if (empty($account)) {
return $this->json(6001, '未登录');
}
CConfig::load('extra/commission_withdrawal', 'commission');
// 提现最低限制
$config = config('commission');
$limitCount = $config['limit_withdraw_commission'] ?? 0;
if ($commission < $limitCount) {
return $this->json(4000, '提现佣金不能小于'.$limitCount);
}
//换算出 1元 = 多少佣金
try {
//本次提现共计金额
$totalMoney = AccountWithdrawalCommission::convertCommissionOrMoney($commission * 100, 'money');
$totalMoney = $totalMoney / 100;//分转元
if ($account['commission'] < $commission) {
return $this->json(4004, '佣金不足');
}
} catch (RepositoryException $e) {
return $this->json(4000, $e->getMessage());
} catch (Exception $e) {
Log::error('发起提现失败 MSG:'.$e->getMessage().' Line:'.$e->getLine().' File:'.$e->getFile());
return $this->json(5000, '提现失败');
}
//条件满足 开始写入提现
Db::startTrans();
try {
//写入日志
AccountDataLog::log($account['id'],
"提现-扣除佣金",
$commission * -1,
AccountDataLog::TYPE_COMMISSION,
AccountDataLog::ACTION_WITHDRAWAL,
Math::sub($account[AccountDataLog::TYPE_COMMISSION], $commission)
);
$rate = $config['withdrawal_proportion'] ?? [];
//写入提现记录
$insert = [
"account_id" => $account['id'],
"number" => $commission,
"role" => '',
"money" => $totalMoney,
"proportion" => $rate['commission'].":".$rate['money'],
"status" => AccountWithdrawalCommission::$status_default,
"created_at" => date("Y-m-d H:i:s"),
"remarks" => '微信号:'.$wechat
];
AccountWithdrawalCommission::create($insert);
//扣除佣金
$account->save(["commission" => Db::raw('`commission` - '.$commission)]);
Db::commit();
return $this->json();
} catch (Exception $e) {
Db::rollback();
Log::error('发起提现失败 MSG:'.$e->getMessage().' Line:'.$e->getLine().' File:'.$e->getFile());
return $this->json(5000, "发起提现失败");
}
}
/**
* 佣金详情
*
* @return Json
* @throws RepositoryException
*/
public function commission(): Json
{
$page = input('page/d', 1);
$size = input('size/d', 10);
$userId = $this->request->user['user_id'] ?? 0;
$repo = AccountRepository::getInstance();
if (!$account = $repo->findById($userId, ['id', 'commission'])) {
return $this->json(6001, '请先登录');
}
$data = [
'commission' => $account['commission'],
'invite_count' => $repo->getInviteCount($userId),
'commission_total' => $repo->getCommissionTotal($userId),//累计佣金 元
'withdrawal_total' => $repo->getCommissionWithdrawalTotal($userId),//累计提现 元
'withdrawing_total' => $repo->getCommissionWithdrawalIngTotal($userId),//提现中 元
'log' => $repo->getCommissionLog($userId, $page, $size)
];
return $this->json(0, 'success', $data);
}
/**
* 获取佣金兑换金额 单位元
*
* @throws RepositoryException
*/
public function commission2money(): Json
{
$num = input('num/f', 0); //元
CConfig::load('extra/commission_withdrawal', 'commission_withdrawal');
$config = config('commission_withdrawal')['withdrawal_proportion'] ?? [];
$res['money'] = Math::fen2Yuan(AccountWithdrawalCommission::convertCommissionOrMoney($num * 100, 'money'));
$res['rate']['commission'] = $config['commission'] ?? '比例未设置';
$res['rate']['money'] = $config['money'] ?? '比例未设置';
return $this->json(0, 'success', $res);
}
}