截流自动化的商城平台
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.

DistributionLevelLogic.php 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514
  1. <?php
  2. namespace app\admin\logic\distribution;
  3. use app\common\basics\Logic;
  4. use app\common\model\distribution\Distribution;
  5. use app\common\model\distribution\DistributionGoods;
  6. use app\common\model\distribution\DistributionLevel;
  7. use app\common\model\distribution\DistributionLevelUpdate;
  8. use app\common\model\distribution\DistributionOrderGoods;
  9. use app\common\model\order\Order;
  10. use app\common\model\order\OrderTrade;
  11. use think\facade\Db;
  12. class DistributionLevelLogic extends Logic
  13. {
  14. /**
  15. * @notes 分销等级列表
  16. * @return array
  17. * @throws \think\db\exception\DataNotFoundException
  18. * @throws \think\db\exception\DbException
  19. * @throws \think\db\exception\ModelNotFoundException
  20. * @author Tab
  21. * @date 2021/9/1 16:19
  22. */
  23. public static function index()
  24. {
  25. $field = [
  26. 'id',
  27. 'name',
  28. 'weights' => 'weights_desc',
  29. 'first_ratio',
  30. 'second_ratio',
  31. 'is_default',
  32. 'id' => 'members_num'
  33. ];
  34. $lists = DistributionLevel::field($field)
  35. ->order('weights', 'asc')
  36. ->select()
  37. ->toArray();
  38. $count = DistributionLevel::count();
  39. return [
  40. 'count' => $count,
  41. 'lists' => $lists,
  42. ];
  43. }
  44. /**
  45. * @notes 添加分销会员等级
  46. * @param $params
  47. * @return bool
  48. * @author Tab
  49. * @date 2021/9/1 14:53
  50. */
  51. public static function add($params)
  52. {
  53. Db::startTrans();
  54. try{
  55. // 写入等级主表
  56. $params['remark'] = $params['remark'] ?? '';
  57. // 佣金比例保留两位小数
  58. $params['first_ratio'] = !empty($params['first_ratio']) ? round($params['first_ratio'], 2) : 0;
  59. $params['second_ratio'] = !empty($params['second_ratio']) ? round($params['second_ratio'], 2) : 0;
  60. $newLevel = DistributionLevel::create($params);
  61. // 写入升级条件表
  62. self::addUpdateCondition($params, $newLevel->id);
  63. // 处理分销商品比例
  64. self::updateDistributionGoods($newLevel->id);
  65. Db::commit();
  66. return true;
  67. }catch(\Exception $e) {
  68. Db::rollback();
  69. self::$error = $e->getMessage();
  70. return false;
  71. }
  72. }
  73. /**
  74. * @notes 添加更新升级条件
  75. * @param $params
  76. * @param $level_id
  77. * @throws \Exception
  78. * @author Tab
  79. * @date 2021/9/1 15:08
  80. */
  81. public static function addUpdateCondition($params, $level_id)
  82. {
  83. $updateConditionData = [];
  84. foreach($params['update_condition'] as $key) {
  85. // 判断是否在规定的条件字段
  86. if(!in_array($key, DistributionLevel::UPDATE_CONDITION_FIELDS, true)) {
  87. continue;
  88. }
  89. if ($params[$key] < 0) {
  90. throw new \Exception('升级条件不允许小于0');
  91. }
  92. // 获取键对应值的字段名
  93. $valueField = DistributionLevel::getValueFiled($key);
  94. $updateConditionData[] = [
  95. 'level_id' => $level_id,
  96. 'key' => $key,
  97. $valueField => $params[$key]
  98. ];
  99. }
  100. (new DistributionLevelUpdate())->saveAll($updateConditionData);
  101. }
  102. /**
  103. * @notes 获取分销等级详情
  104. * @param $params
  105. * @return array
  106. * @author Tab
  107. * @date 2021/9/1 15:36
  108. */
  109. public static function detail($params)
  110. {
  111. $level = DistributionLevel::withoutField('create_time,update_time,delete_time')->findOrEmpty($params['id']);
  112. if($level->isEmpty()) {
  113. return [];
  114. }
  115. $level = $level->toArray();
  116. // 默认等级
  117. if($level['is_default']) {
  118. unset($level['self_ratio']);
  119. unset($level['third_ratio']);
  120. unset($level['update_relation']);
  121. return $level;
  122. }
  123. // 自定义等级
  124. $level['update_condition'] = self::getUpdateCondition($level);
  125. unset($level['self_ratio']);
  126. unset($level['third_ratio']);
  127. return $level;
  128. }
  129. /**
  130. * @notes 获取升级条件
  131. * @param $level
  132. * @return array
  133. * @author Tab
  134. * @date 2021/9/1 15:36
  135. */
  136. public static function getUpdateCondition($level)
  137. {
  138. $updateCondition = DistributionLevelUpdate::where('level_id', $level['id'])->column('key,value_int,value_decimal,value_text');
  139. $updateConditionData = [];
  140. foreach($updateCondition as $item) {
  141. if($item['value_int']) {
  142. $updateConditionData[$item['key']] = $item['value_int'];
  143. continue;
  144. }
  145. if($item['value_decimal']) {
  146. $updateConditionData[$item['key']] = $item['value_decimal'];
  147. continue;
  148. }
  149. if($item['value_text']) {
  150. $updateConditionData[$item['key']] = $item['value_text'];
  151. continue;
  152. }
  153. }
  154. $data = [
  155. 'keys' => array_keys($updateConditionData),
  156. 'data' => $updateConditionData
  157. ];
  158. // 补全条件
  159. foreach(DistributionLevel::UPDATE_CONDITION_FIELDS as $field) {
  160. if(!isset($data['data'][$field])) {
  161. $data['data'][$field] = '';
  162. }
  163. }
  164. return $data;
  165. }
  166. /**
  167. * @notes 编辑分销等级
  168. * @param $params
  169. * @return bool
  170. * @author Tab
  171. * @date 2021/9/1 16:20
  172. */
  173. public static function edit($params)
  174. {
  175. Db::startTrans();
  176. try{
  177. $params['remark'] = $params['remark'] ?? '';
  178. $level = DistributionLevel::findOrEmpty($params['id']);
  179. if($level->isEmpty()) {
  180. throw new \Exception('等级不存在');
  181. }
  182. // 佣金比例保留两位小数
  183. $params['first_ratio'] = !empty($params['first_ratio']) ? round($params['first_ratio'], 2) : 0;
  184. $params['second_ratio'] = !empty($params['second_ratio']) ? round($params['second_ratio'], 2) : 0;
  185. // 默认等级
  186. if($level->is_default) {
  187. $level->allowField(['name', 'first_ratio', 'second_ratio','remark'])->save($params);
  188. Db::commit();
  189. return true;
  190. }
  191. // 自定义等级 - 更新主表信息
  192. if(!$params['weights'] > 1) {
  193. throw new \Exception('级别须大于1');
  194. }
  195. if(!isset($params['update_relation'])) {
  196. throw new \Exception('请选择升级关系');
  197. }
  198. if(!isset($params['update_condition']) || !count($params['update_condition'])) {
  199. throw new \Exception('请选择升级条件');
  200. }
  201. $level->allowField(['name', 'weights', 'first_ratio', 'second_ratio','remark', 'update_relation'])->save($params);
  202. // 自定义等级 - 删除旧升级条件
  203. $deleteIds = DistributionLevelUpdate::where('level_id', $level->id)->column('id');
  204. DistributionLevelUpdate::destroy($deleteIds);
  205. // 自定义等级 - 添加新的升级条件
  206. self::addUpdateCondition($params, $level->id);
  207. Db::commit();
  208. return true;
  209. }catch(\Exception $e) {
  210. Db::rollback();
  211. self::$error = $e->getMessage();
  212. return false;
  213. }
  214. }
  215. /**
  216. * @notes 删除分销等级
  217. * @param $params
  218. * @return bool
  219. * @author Tab
  220. * @date 2021/9/1 16:21
  221. */
  222. public static function delete($params)
  223. {
  224. Db::startTrans();
  225. try{
  226. $level = DistributionLevel::findOrEmpty($params['id']);
  227. if($level->isEmpty()) {
  228. throw new \Exception('等级不存在');
  229. }
  230. if($level->is_default) {
  231. throw new \Exception('系统默认等级不允许删除');
  232. }
  233. // 重置该等级下的分销会员为系统默认等级
  234. $defaultId = DistributionLevel::where('is_default', 1)->value('id');
  235. Distribution::where('level_id', $level->id)->update(['level_id' => $defaultId]);
  236. // 删除升级条件
  237. $deleteIds = DistributionLevelUpdate::where('level_id', $level->id)->column('id');
  238. DistributionLevelUpdate::destroy($deleteIds);
  239. // 删除该等级下的分销商品比例
  240. $deleteIds = DistributionGoods::where('level_id', $level->id)->column('id');
  241. DistributionGoods::destroy($deleteIds);
  242. // 删除等级
  243. $level->delete();
  244. Db::commit();
  245. return true;
  246. }catch(\Exception $e) {
  247. Db::rollback();
  248. self::$error = $e->getMessage();
  249. return false;
  250. }
  251. }
  252. /**
  253. * @notes 更新分销会员等级
  254. * @param $userId
  255. * @return false
  256. * @author Tab
  257. * @date 2021/9/2 16:39
  258. */
  259. public static function updateDistributionLevel($userId)
  260. {
  261. // 非默认等级
  262. $levels = DistributionLevel::where('is_default', 0)
  263. ->order('weights', 'desc')
  264. ->column('id,name,weights,update_relation', 'id');
  265. $userInfo = Distribution::alias('d')
  266. ->leftJoin('distribution_level dl', 'dl.id = d.level_id')
  267. ->field('d.is_distribution,d.level_id,dl.weights')
  268. ->where('d.user_id', $userId)
  269. ->findOrEmpty()
  270. ->toArray();
  271. // 非分销会员直接返回false
  272. if(empty($userInfo) || !$userInfo['is_distribution']) {
  273. return false;
  274. }
  275. foreach($levels as $level) {
  276. if(self::isMeetConditions($userId, $level) && $level['weights'] > $userInfo['weights']) {
  277. // 满足升级条件且是升更高的等级
  278. Distribution::where(['user_id' => $userId])->update(['level_id' => $level['id']]);
  279. break;
  280. }
  281. }
  282. }
  283. /**
  284. * @notes 判断是否满足当前等级的升级条件
  285. * @param $userId
  286. * @param $level
  287. * @return bool
  288. * @author Tab
  289. * @date 2021/9/2 16:42
  290. */
  291. public static function isMeetConditions($userId, $level)
  292. {
  293. $updateRelation = $level['update_relation'];
  294. // 任一条件满足升级
  295. if($updateRelation == 1) {
  296. $flagOr = self::singleConsumptionAmountFlag($userId, $level, $updateRelation)
  297. || self::cumulativeConsumptionAmountFlag($userId, $level, $updateRelation)
  298. || self::cumulativeConsumptionTimesFlag($userId, $level, $updateRelation)
  299. || self::returnedCommissionFlag($userId, $level, $updateRelation);
  300. return $flagOr;
  301. }
  302. // 全部条件满足升级
  303. if($updateRelation == 2) {
  304. $flagAnd = self::singleConsumptionAmountFlag($userId, $level, $updateRelation)
  305. && self::cumulativeConsumptionAmountFlag($userId, $level, $updateRelation)
  306. && self::cumulativeConsumptionTimesFlag($userId, $level, $updateRelation)
  307. && self::returnedCommissionFlag($userId, $level, $updateRelation);
  308. return $flagAnd;
  309. }
  310. }
  311. /**
  312. * @notes 判断是否满足单笔消费金额条件
  313. * @param $userId
  314. * @param $level
  315. * @return bool
  316. * @author Tab
  317. * @date 2021/9/2 16:44
  318. */
  319. public static function singleConsumptionAmountFlag($userId, $level, $updateRelation)
  320. {
  321. $condition = DistributionLevelUpdate::field('value_int,value_decimal,value_text')
  322. ->where([
  323. 'level_id' => $level['id'],
  324. 'key' => 'singleConsumptionAmount'
  325. ])
  326. ->findOrEmpty();
  327. if($condition->isEmpty()) {
  328. // 等级条件为满足任一条件(updateRelation=1) 返回false (满足已设置的任一条件时才升级,未设置的条件归为未满足,返回false)
  329. // 等级条件为满足全部条件(updateRelation=2) 返回true (满足已设置的所有条件时才升级,未设置的条件归为已满足,返回true)
  330. return $updateRelation == 2;
  331. }
  332. $recentOrder = Order::where([
  333. 'user_id' => $userId,
  334. 'pay_status' => 1
  335. ])
  336. ->order('id', 'desc')
  337. ->findOrEmpty();
  338. if($recentOrder->isEmpty()) {
  339. return false;
  340. }
  341. $singleConsumptionAmount = OrderTrade::where('id', $recentOrder['trade_id'])->findOrEmpty()->toArray();
  342. if($singleConsumptionAmount['order_amount'] >= $condition['value_decimal']) {
  343. return true;
  344. }
  345. return false;
  346. }
  347. /**
  348. * @notes 判断是否满足累计消费金额条件
  349. * @param $userId
  350. * @param $level
  351. * @return bool
  352. * @author Tab
  353. * @date 2021/9/2 16:50
  354. */
  355. public static function cumulativeConsumptionAmountFlag($userId, $level, $updateRelation)
  356. {
  357. $condition = DistributionLevelUpdate::field('value_int,value_decimal,value_text')
  358. ->where([
  359. 'level_id' => $level['id'],
  360. 'key' => 'cumulativeConsumptionAmount'
  361. ])
  362. ->findOrEmpty();
  363. if($condition->isEmpty()) {
  364. // 等级条件为满足任一条件(updateRelation=1) 返回false (满足已设置的任一条件时才升级,未设置的条件归为未满足,返回false)
  365. // 等级条件为满足全部条件(updateRelation=2) 返回true (满足已设置的所有条件时才升级,未设置的条件归为已满足,返回true)
  366. return $updateRelation == 2;
  367. }
  368. $cumulativeConsumptionAmount = Order::where([
  369. 'user_id' => $userId,
  370. 'pay_status' => 1
  371. ])->sum('order_amount');
  372. if($cumulativeConsumptionAmount >= $condition['value_decimal']) {
  373. return true;
  374. }
  375. return false;
  376. }
  377. /**
  378. * @notes 判断是否满足累计消费次数条件
  379. * @param $userId
  380. * @param $level
  381. * @return bool
  382. * @author Tab
  383. * @date 2021/9/2 16:53
  384. */
  385. public static function cumulativeConsumptionTimesFlag($userId, $level, $updateRelation)
  386. {
  387. $condition = DistributionLevelUpdate::field('value_int,value_decimal,value_text')
  388. ->where([
  389. 'level_id' => $level['id'],
  390. 'key' => 'cumulativeConsumptionTimes'
  391. ])
  392. ->findOrEmpty();
  393. if($condition->isEmpty()) {
  394. // 等级条件为满足任一条件(updateRelation=1) 返回false (满足已设置的任一条件时才升级,未设置的条件归为未满足,返回false)
  395. // 等级条件为满足全部条件(updateRelation=2) 返回true (满足已设置的所有条件时才升级,未设置的条件归为已满足,返回true)
  396. return $updateRelation == 2;
  397. }
  398. $allTradeIds = Order::where([
  399. 'user_id' => $userId,
  400. 'pay_status' => 1
  401. ])->column('trade_id');
  402. $cumulativeConsumptionTimes = count(array_unique($allTradeIds));
  403. if($cumulativeConsumptionTimes >= $condition['value_int']) {
  404. return true;
  405. }
  406. return false;
  407. }
  408. /**
  409. * @notes 判断是否消费已返佣金条件
  410. * @param $userId
  411. * @param $level
  412. * @return bool
  413. * @author Tab
  414. * @date 2021/9/2 17:06
  415. */
  416. public static function returnedCommissionFlag($userId, $level, $updateRelation)
  417. {
  418. $condition = DistributionLevelUpdate::field('value_int,value_decimal,value_text')
  419. ->where([
  420. 'level_id' => $level['id'],
  421. 'key' => 'returnedCommission'
  422. ])
  423. ->findOrEmpty();
  424. if($condition->isEmpty()) {
  425. // 等级条件为满足任一条件(updateRelation=1) 返回false (满足已设置的任一条件时才升级,未设置的条件归为未满足,返回false)
  426. // 等级条件为满足全部条件(updateRelation=2) 返回true (满足已设置的所有条件时才升级,未设置的条件归为已满足,返回true)
  427. return $updateRelation == 2;
  428. }
  429. $returnedCommission = DistributionOrderGoods::where([
  430. 'user_id' => $userId,
  431. 'status' => 2
  432. ])->sum('money');
  433. if($returnedCommission >= $condition['value_decimal']) {
  434. return true;
  435. }
  436. return false;
  437. }
  438. /**
  439. * @notes 获取所有分销会员等级
  440. * @return array
  441. * @author Tab
  442. * @date 2021/9/2 18:31
  443. */
  444. public static function getLevels()
  445. {
  446. return DistributionLevel::order('weights', 'asc')->column('id, name,weights');
  447. }
  448. /**
  449. * @notes 更新分销商品比例
  450. * @param $levelId
  451. * @author Tab
  452. * @date 2021/9/7 17:27
  453. */
  454. public static function updateDistributionGoods($levelId)
  455. {
  456. // 处理单独设置比例的商品,新增分销会等级佣金比例初始化为0
  457. $field = [
  458. 'shop_id',
  459. 'goods_id',
  460. 'item_id',
  461. ];
  462. $distribution = DistributionGoods::distinct(true)->field($field)->where('rule', 2)->select()->toArray();
  463. $addData = [];
  464. foreach($distribution as $item) {
  465. $temp = [
  466. 'shop_id' => $item['shop_id'],
  467. 'goods_id' => $item['goods_id'],
  468. 'item_id' => $item['item_id'],
  469. 'level_id' => $levelId,
  470. 'first_ratio' => 0,
  471. 'second_ratio' => 0,
  472. 'rule' => 2,
  473. ];
  474. $addData[] = $temp;
  475. }
  476. (new DistributionGoods())->saveAll($addData);
  477. }
  478. }