PrizeService.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. <?php
  2. namespace app\data\service;
  3. use think\admin\Service;
  4. use think\admin\Exception;
  5. /**
  6. * 用户奖励配置
  7. * Class PrizeService
  8. * @package app\data\service
  9. */
  10. class PrizeService extends Service
  11. {
  12. const PRIZE_01 = 'prize_01';
  13. const PRIZE_02 = 'prize_02';
  14. const PRIZE_03 = 'prize_03';
  15. const PRIZE_04 = 'prize_04';
  16. const PRIZE_05 = 'prize_05';
  17. const PRIZES = [
  18. self::PRIZE_01 => ['code' => self::PRIZE_01, 'name' => '首推奖励', 'func' => '_prize01'],
  19. self::PRIZE_02 => ['code' => self::PRIZE_02, 'name' => '复购奖励', 'func' => '_prize02'],
  20. self::PRIZE_03 => ['code' => self::PRIZE_03, 'name' => '直属团队', 'func' => '_prize03'],
  21. self::PRIZE_04 => ['code' => self::PRIZE_04, 'name' => '间接团队', 'func' => '_prize04'],
  22. self::PRIZE_05 => ['code' => self::PRIZE_05, 'name' => '差额奖励', 'func' => '_prize05'],
  23. ];
  24. /**
  25. * 用户数据
  26. * @var array
  27. */
  28. protected $user;
  29. /**
  30. * 订单数据
  31. * @var array
  32. */
  33. protected $order;
  34. /**
  35. * 推荐用户
  36. * @var array
  37. */
  38. protected $fromer;
  39. /**
  40. * 绑定数据表
  41. * @var string
  42. */
  43. private $table = 'DataUserRebate';
  44. /**
  45. * 获取奖励名称
  46. * @param string $prize
  47. * @return string
  48. */
  49. public function name(string $prize): string
  50. {
  51. return self::PRIZES[$prize]['name'] ?? $prize;
  52. }
  53. /**
  54. * 执行订单返利
  55. * @param string $orderNo
  56. * @throws Exception
  57. * @throws \think\db\exception\DataNotFoundException
  58. * @throws \think\db\exception\DbException
  59. * @throws \think\db\exception\ModelNotFoundException
  60. */
  61. public function execute(string $orderNo)
  62. {
  63. // 获取订单数据
  64. $map = ['order_no' => $orderNo, 'payment_status' => 1];
  65. $this->order = $this->app->db->name('ShopOrder')->where($map)->find();
  66. if (empty($this->order)) throw new Exception('订单不存在');
  67. // 获取用户数据
  68. $map = ['id' => $this->order['uid']];
  69. $this->user = $this->app->db->name('DataUser')->where($map)->find();
  70. if (empty($this->user)) throw new Exception('用户不存在');
  71. // 获取推荐用户
  72. if ($this->order['from'] > 0) {
  73. $map = ['id' => $this->order['from']];
  74. $this->fromer = $this->app->db->name('DataUser')->where($map)->find();
  75. if (empty($this->fromer)) throw new Exception('推荐不存在');
  76. }
  77. // 批量发放奖励
  78. foreach (self::PRIZES as $vo) {
  79. if (method_exists($this, $vo['func'])) {
  80. $this->{$vo['func']}();
  81. }
  82. }
  83. }
  84. /**
  85. * 首推奖励
  86. * @return boolean
  87. * @throws \think\db\exception\DataNotFoundException
  88. * @throws \think\db\exception\DbException
  89. * @throws \think\db\exception\ModelNotFoundException
  90. */
  91. protected function _prize01(): bool
  92. {
  93. if (empty($this->fromer)) return false;
  94. $map = ['order_uid' => $this->user['id']];
  95. if ($this->app->db->name($this->table)->where($map)->count() > 0) return false;
  96. if (!$this->checkLevelPrize(self::PRIZE_01, $this->fromer['vip_number'])) return false;
  97. // 创建返利奖励记录
  98. $map = ['type' => self::PRIZE_01, 'order_no' => $this->order['order_no'], 'order_uid' => $this->order['uid']];
  99. if ($this->app->db->name($this->table)->where($map)->count() < 1) {
  100. if (sysconf('shop.fristType') == 1) {
  101. $amount = sysconf('shop.fristValue') ?: '0.00';
  102. $name = self::instance()->name(self::PRIZE_01) . ",每人 {$amount} 元";
  103. } else {
  104. $amount = sysconf('shop.fristValue') * $this->order['amount_total'] / 100;
  105. $name = self::instance()->name(self::PRIZE_01) . ",订单 " . sysconf('shop.fristValue') . '%';
  106. }
  107. $this->app->db->name($this->table)->insert(array_merge($map, [
  108. 'uid' => $this->fromer['id'], 'name' => $name, 'amount' => $amount, 'order_amount' => $this->order['amount_total'],
  109. ]));
  110. // 更新用户奖利金额
  111. UserService::instance()->syncLevel($this->fromer['id']);
  112. }
  113. return true;
  114. }
  115. /**
  116. * 复购奖励
  117. * @return boolean
  118. * @throws \think\db\exception\DataNotFoundException
  119. * @throws \think\db\exception\DbException
  120. * @throws \think\db\exception\ModelNotFoundException
  121. */
  122. protected function _prize02(): bool
  123. {
  124. if (empty($this->fromer)) return false;
  125. $map = ['order_uid' => $this->user['id']];
  126. if ($this->app->db->name($this->table)->where($map)->count() < 1) return false;
  127. if (!$this->checkLevelPrize(self::PRIZE_02, $this->fromer['vip_number'])) return false;
  128. // 创建返利奖励记录
  129. $map = ['type' => self::PRIZE_02, 'order_no' => $this->order['order_no'], 'order_uid' => $this->order['uid']];
  130. if ($this->app->db->name($this->table)->where($map)->count() < 1) {
  131. if (sysconf('shop.repeatType') == 1) {
  132. $amount = sysconf('shop.repeatValue') ?: '0.00';
  133. $name = self::instance()->name(self::PRIZE_02) . ",每人 {$amount} 元";
  134. } else {
  135. $amount = sysconf('shop.repeatValue') * $this->order['amount_total'] / 100;
  136. $name = self::instance()->name(self::PRIZE_02) . ",订单 " . sysconf('shop.repeatValue') . '%';
  137. }
  138. $this->app->db->name($this->table)->insert(array_merge($map, [
  139. 'uid' => $this->fromer['id'], 'name' => $name, 'amount' => $amount, 'order_amount' => $this->order['amount_total'],
  140. ]));
  141. // 更新用户奖利金额
  142. UserService::instance()->syncLevel($this->fromer['id']);
  143. }
  144. return true;
  145. }
  146. /**
  147. * 直属团队
  148. * @return boolean
  149. * @throws \think\db\exception\DataNotFoundException
  150. * @throws \think\db\exception\DbException
  151. * @throws \think\db\exception\ModelNotFoundException
  152. */
  153. private function _prize03(): bool
  154. {
  155. if (empty($this->fromer)) return false;
  156. if (!$this->checkLevelPrize(self::PRIZE_03, $this->fromer['vip_number'])) return false;
  157. // 创建返利奖励记录
  158. $map = ['type' => self::PRIZE_03, 'order_no' => $this->order['order_no'], 'order_uid' => $this->order['uid']];
  159. if ($this->app->db->name($this->table)->where($map)->count() < 1) {
  160. $amount = sysconf('shop.repeatValue') * $this->order['amount_total'] / 100;
  161. $name = self::instance()->name(self::PRIZE_03) . ",订单 " . sysconf('shop.repeatValue') . '%';
  162. $this->app->db->name($this->table)->insert(array_merge($map, [
  163. 'uid' => $this->fromer['id'], 'name' => $name, 'amount' => $amount, 'order_amount' => $this->order['amount_total'],
  164. ]));
  165. // 更新用户奖利金额
  166. UserService::instance()->syncLevel($this->fromer['id']);
  167. }
  168. return true;
  169. }
  170. /**
  171. * 间接团队
  172. * @return boolean
  173. * @throws \think\db\exception\DataNotFoundException
  174. * @throws \think\db\exception\DbException
  175. * @throws \think\db\exception\ModelNotFoundException
  176. */
  177. private function _prize04(): bool
  178. {
  179. if (empty($this->fromer) || empty($this->fromer['pid1'])) return false;
  180. $pm2 = $this->app->db->name('DataUser')->where(['id' => $this->fromer['pid1']])->find();
  181. if (empty($pm2)) return false;
  182. if (!$this->checkLevelPrize(self::PRIZE_04, $pm2['vip_number'])) return false;
  183. $map = ['type' => self::PRIZE_04, 'order_no' => $this->order['order_no'], 'order_uid' => $this->order['uid']];
  184. if ($this->app->db->name($this->table)->where($map)->count() < 1) {
  185. $amount = sysconf('shop.indirectValue') * $this->order['amount_total'] / 100;
  186. $name = self::instance()->name(self::PRIZE_04) . ",订单 " . sysconf('shop.indirectValue') . '%';
  187. $this->app->db->name($this->table)->insert(array_merge($map, [
  188. 'uid' => $pm2['id'], 'name' => $name, 'amount' => $amount, 'order_amount' => $this->order['amount_total'],
  189. ]));
  190. // 更新代理奖利金额
  191. UserService::instance()->syncLevel($pm2['id']);
  192. }
  193. return true;
  194. }
  195. /**
  196. * 差额奖励
  197. * @return false
  198. * @throws \think\db\exception\DataNotFoundException
  199. * @throws \think\db\exception\DbException
  200. * @throws \think\db\exception\ModelNotFoundException
  201. */
  202. private function _prize05(): bool
  203. {
  204. $pids = array_reverse(explode('-', trim($this->user['path'], '-')));
  205. if (empty($pids)) return false;
  206. // 获取拥有差额奖励的等级
  207. $query = $this->app->db->name('DataUserLevel');
  208. $numbs = $query->whereLike('rule', '%,' . self::PRIZE_05 . ',%')->column('number');
  209. // 获取可以参与奖励的代理
  210. $query = $this->app->db->name('DataUser')->whereIn('vip_number', $numbs);
  211. $users = $query->whereIn('id', $pids)->orderField('id', $pids)->select()->toArray();
  212. // 查询需要计算奖励的商品
  213. $map = [['order_no', '=', $this->order['order_no']], ['discount_rate', '<', 100]];
  214. $this->app->db->name('StoreOrderItem')->where($map)->select()->each(function ($item) use ($users) {
  215. $itemJson = $this->app->db->name('DataUserDiscount')->where(['status' => 1, 'deleted' => 0])->value('items');
  216. if (!empty($itemJson) && is_array($rules = json_decode($itemJson, true))) {
  217. [$tVip, $tRate] = [$item['vip_number'], $item['discount_rate']];
  218. foreach ($rules as $rule) if ($rule['level'] > $tVip) foreach ($users as $user) if ($user['vip_number'] > $tVip) {
  219. if ($tRate > $rule['discount'] && $tRate < 100) {
  220. $map = [
  221. 'uid' => $user['id'],
  222. 'type' => self::PRIZE_05,
  223. 'code' => "{$this->order['order_no']}#{$item['id']}#{$tVip}.{$user['vip_number']}",
  224. ];
  225. if ($this->app->db->name($this->table)->where($map)->count() < 1) {
  226. $dRate = ($tRate - $rule['discount']) / 100;
  227. $this->app->db->name($this->table)->insert(array_merge($map, [
  228. 'name' => "等级差额奖励{$tVip}#{$user['vip_number']}商品的{$dRate}%",
  229. 'amount' => $dRate * $item['total_selling'],
  230. 'order_no' => $this->order['order_no'],
  231. 'order_uid' => $this->order['uid'],
  232. 'order_amount' => $this->order['amount_total'],
  233. ]));
  234. }
  235. [$tVip, $tRate] = [$user['vip_number'], $rule['discount']];
  236. }
  237. }
  238. }
  239. });
  240. return true;
  241. }
  242. /**
  243. * 检查等级是否有奖励
  244. * @param string $prize
  245. * @param integer $level
  246. * @return boolean
  247. */
  248. protected function checkLevelPrize(string $prize, int $level): bool
  249. {
  250. $map = [['number', '=', $level], ['rebate_rule', 'like', "%,{$prize},%"]];
  251. return $this->app->db->name('DataUserLevel')->where($map)->count() > 0;
  252. }
  253. }