| New file | 
|  |  |  | 
|---|
|  |  |  | package com.ruoyi.account.service.impl; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import cn.hutool.core.collection.CollectionUtil; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.core.metadata.IPage; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.extension.conditions.query.LambdaQueryChainWrapper; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.extension.conditions.query.QueryChainWrapper; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; | 
|---|
|  |  |  | import com.ruoyi.account.api.model.AppUser; | 
|---|
|  |  |  | import com.ruoyi.account.api.model.UserPoint; | 
|---|
|  |  |  | import com.ruoyi.account.mapper.UserPointMapper; | 
|---|
|  |  |  | import com.ruoyi.account.service.AppUserService; | 
|---|
|  |  |  | import com.ruoyi.account.service.PointSettingService; | 
|---|
|  |  |  | import com.ruoyi.account.service.UserPointService; | 
|---|
|  |  |  | import com.ruoyi.account.service.VipSettingService; | 
|---|
|  |  |  | import com.ruoyi.account.vo.UserPointDetailVO; | 
|---|
|  |  |  | import com.ruoyi.account.vo.UserPointStatistics; | 
|---|
|  |  |  | import com.ruoyi.account.vo.UserPointVO; | 
|---|
|  |  |  | import com.ruoyi.common.core.domain.R; | 
|---|
|  |  |  | import com.ruoyi.common.core.exception.ServiceException; | 
|---|
|  |  |  | import com.ruoyi.common.core.utils.PhoneNumberValidator; | 
|---|
|  |  |  | import com.ruoyi.common.core.web.page.PageInfo; | 
|---|
|  |  |  | import com.ruoyi.common.security.service.TokenService; | 
|---|
|  |  |  | import com.ruoyi.other.api.domain.PointSetting; | 
|---|
|  |  |  | import com.ruoyi.other.api.domain.VipSetting; | 
|---|
|  |  |  | import com.ruoyi.other.api.enums.PointChangeType; | 
|---|
|  |  |  | import com.ruoyi.system.api.model.LoginUser; | 
|---|
|  |  |  | import lombok.extern.slf4j.Slf4j; | 
|---|
|  |  |  | import org.springframework.stereotype.Service; | 
|---|
|  |  |  | import org.springframework.transaction.annotation.Transactional; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import javax.annotation.Resource; | 
|---|
|  |  |  | import java.math.BigDecimal; | 
|---|
|  |  |  | import java.time.LocalDateTime; | 
|---|
|  |  |  | import java.time.format.DateTimeFormatter; | 
|---|
|  |  |  | import java.util.Collections; | 
|---|
|  |  |  | import java.util.List; | 
|---|
|  |  |  | import java.util.Map; | 
|---|
|  |  |  | import java.util.stream.Collectors; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * <p> | 
|---|
|  |  |  | * 服务实现类 | 
|---|
|  |  |  | * </p> | 
|---|
|  |  |  | * | 
|---|
|  |  |  | * @author luodangjia | 
|---|
|  |  |  | * @since 2024-11-21 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | @Service | 
|---|
|  |  |  | @Slf4j | 
|---|
|  |  |  | public class UserPointServiceImpl extends ServiceImpl<UserPointMapper, UserPoint> implements UserPointService { | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private AppUserService appUserService; | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private TokenService tokenService; | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private VipSettingService vipSettingService; | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private PointSettingService pointSettingService; | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private UserPointService userPointService; | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public UserPointVO getUserPoint(Long userId) { | 
|---|
|  |  |  | AppUser appUser = appUserService.getById(userId); | 
|---|
|  |  |  | VipSetting vipSetting = vipSettingService.getVipSettingByUserId(userId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | UserPointVO userPointVO = new UserPointVO(); | 
|---|
|  |  |  | // todo bug说取剩余积分 | 
|---|
|  |  |  | userPointVO.setTotalPoint(appUser.getLavePoint()); | 
|---|
|  |  |  | userPointVO.setConsumePoint(appUser.getAvailablePoint()); | 
|---|
|  |  |  | userPointVO.setShopPoint(appUser.getShopPoint()); | 
|---|
|  |  |  | userPointVO.setSharePoint(appUser.getSharePoint()); | 
|---|
|  |  |  | userPointVO.setPullNewPoint(appUser.getTotalInvitePoint()); | 
|---|
|  |  |  | userPointVO.setShopAchievementPoint(appUser.getShopAchievementPoint()); | 
|---|
|  |  |  | userPointVO.setShopSharePoint(appUser.getShopSharePoint()); | 
|---|
|  |  |  | userPointVO.setGiftPoint(vipSetting.getVipGiftRole() == 1 ? 1 : 0); | 
|---|
|  |  |  | return userPointVO; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public PageInfo<UserPointDetailVO> getUserPointDetail(Long userId, LocalDateTime startTime, LocalDateTime endTime, Integer type, Integer pageCurr, Integer pageSize) { | 
|---|
|  |  |  | PageInfo<UserPoint> pageInfo = new PageInfo<>(pageCurr, pageSize); | 
|---|
|  |  |  | PageInfo<UserPoint> page = this.page(pageInfo, new QueryWrapper<UserPoint>() | 
|---|
|  |  |  | .between(startTime != null, "create_time", startTime, endTime) | 
|---|
|  |  |  | .eq(type != null, "type", type) | 
|---|
|  |  |  | .eq("app_user_id", userId).orderByDesc("create_time")); | 
|---|
|  |  |  | if (CollectionUtil.isNotEmpty(page.getRecords())) { | 
|---|
|  |  |  | List<UserPointDetailVO> collect = page.getRecords().stream().map(p -> { | 
|---|
|  |  |  | UserPointDetailVO userPointDetailVO = new UserPointDetailVO(); | 
|---|
|  |  |  | userPointDetailVO.setType(p.getType()); | 
|---|
|  |  |  | userPointDetailVO.setVariablePoint(p.getVariablePoint()); | 
|---|
|  |  |  | String format = p.getCreateTime().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); | 
|---|
|  |  |  | userPointDetailVO.setCreateTime(format); | 
|---|
|  |  |  | Integer historicalPoint = p.getHistoricalPoint(); | 
|---|
|  |  |  | Integer balance = p.getBalance(); | 
|---|
|  |  |  | if (historicalPoint != null && balance != null) { | 
|---|
|  |  |  | userPointDetailVO.setFlag(historicalPoint > balance ? 2 : 1); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return userPointDetailVO; | 
|---|
|  |  |  | }).collect(Collectors.toList()); | 
|---|
|  |  |  | PageInfo<UserPointDetailVO> pageInfo1 = new PageInfo<>(pageCurr, pageSize); | 
|---|
|  |  |  | pageInfo1.setRecords(collect); | 
|---|
|  |  |  | pageInfo1.setTotal(page.getTotal()); | 
|---|
|  |  |  | pageInfo1.setCurrent(page.getCurrent()); | 
|---|
|  |  |  | pageInfo1.setSize(page.getSize()); | 
|---|
|  |  |  | return pageInfo1; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | @Transactional | 
|---|
|  |  |  | public R transferPoint(Integer point, String phone) { | 
|---|
|  |  |  | if (!PhoneNumberValidator.isValidChinaPhoneNumber(phone)) { | 
|---|
|  |  |  | return R.fail("无效的电话号码"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | LoginUser loginUserApplet = tokenService.getLoginUserApplet(); | 
|---|
|  |  |  | Long userid = loginUserApplet.getUserid(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | VipSetting vipSetting = vipSettingService.getVipSettingByUserId(userid); | 
|---|
|  |  |  | if (vipSetting == null) { | 
|---|
|  |  |  | return R.fail("VIP 设置未找到"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (vipSetting.getVipGiftRole() == 0) { | 
|---|
|  |  |  | return R.fail("转赠积分权限未开启"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | AppUser appUser = appUserService.getById(userid); | 
|---|
|  |  |  | if (appUser == null) { | 
|---|
|  |  |  | return R.fail("用户未找到"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 可转赠积分总数 | 
|---|
|  |  |  | Integer transferablePoint = appUser.getTransferablePoint(); | 
|---|
|  |  |  | if (point > transferablePoint) { | 
|---|
|  |  |  | return R.fail("可转赠积分不足"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | AppUser appUserForPhoe = appUserService.getOne(new LambdaQueryWrapper<AppUser>() | 
|---|
|  |  |  | .eq(AppUser::getPhone, phone).eq(AppUser::getDelFlag, 0).ne(AppUser::getStatus, 3)); | 
|---|
|  |  |  | if (appUserForPhoe == null) { | 
|---|
|  |  |  | return R.fail("目标用户不存在"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if(appUserForPhoe.getStatus() == 2){ | 
|---|
|  |  |  | return R.fail("目标用户已被冻结"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if(appUserForPhoe.getId().equals(userid)){ | 
|---|
|  |  |  | return R.fail("不能转增给自己"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Integer lavePoint1 = appUserForPhoe.getLavePoint(); | 
|---|
|  |  |  | appUserForPhoe.setLavePoint(appUserForPhoe.getLavePoint() + point); | 
|---|
|  |  |  | appUserForPhoe.setTotalPoint(appUserForPhoe.getTotalPoint() + point); | 
|---|
|  |  |  | appUserForPhoe.setTransferablePoint(transferablePoint + point); | 
|---|
|  |  |  | appUserForPhoe.setAvailablePoint(appUserForPhoe.getAvailablePoint() + point); | 
|---|
|  |  |  | appUserForPhoe.setTotalAvailablePoint(appUserForPhoe.getTotalAvailablePoint() + point); | 
|---|
|  |  |  | appUserService.updateById(appUserForPhoe); | 
|---|
|  |  |  | //构建积分流水记录 | 
|---|
|  |  |  | UserPoint userPoint = new UserPoint(); | 
|---|
|  |  |  | userPoint.setType(12); | 
|---|
|  |  |  | userPoint.setHistoricalPoint(lavePoint1); | 
|---|
|  |  |  | userPoint.setVariablePoint(point); | 
|---|
|  |  |  | userPoint.setBalance(appUserForPhoe.getLavePoint()); | 
|---|
|  |  |  | userPoint.setCreateTime(LocalDateTime.now()); | 
|---|
|  |  |  | userPoint.setAppUserId(appUserForPhoe.getId()); | 
|---|
|  |  |  | userPointService.save(userPoint); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Integer lavePoint = appUser.getLavePoint(); | 
|---|
|  |  |  | appUser.setLavePoint(appUser.getLavePoint() - point); | 
|---|
|  |  |  | appUser.setTransferablePoint(appUser.getTransferablePoint() - point); | 
|---|
|  |  |  | appUser.setAvailablePoint(appUser.getAvailablePoint() - point); | 
|---|
|  |  |  | appUserService.updateById(appUser); | 
|---|
|  |  |  | //构建积分流水记录 | 
|---|
|  |  |  | userPoint = new UserPoint(); | 
|---|
|  |  |  | userPoint.setType(13); | 
|---|
|  |  |  | userPoint.setHistoricalPoint(lavePoint); | 
|---|
|  |  |  | userPoint.setVariablePoint(point); | 
|---|
|  |  |  | userPoint.setBalance(appUser.getLavePoint()); | 
|---|
|  |  |  | userPoint.setCreateTime(LocalDateTime.now()); | 
|---|
|  |  |  | userPoint.setAppUserId(appUser.getId()); | 
|---|
|  |  |  | userPointService.save(userPoint); | 
|---|
|  |  |  | return R.ok(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public UserPointStatistics getStatistics(UserPoint userPoint) { | 
|---|
|  |  |  | List<UserPoint> userPointList = this.baseMapper.selectUserPoint(userPoint); | 
|---|
|  |  |  | Map<Integer, Integer> userBalanceMap = userPointList.stream() | 
|---|
|  |  |  | .collect(Collectors.groupingBy( | 
|---|
|  |  |  | UserPoint::getType, | 
|---|
|  |  |  | Collectors.summingInt(UserPoint::getVariablePoint) | 
|---|
|  |  |  | )); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int consumePoint = userBalanceMap.getOrDefault(PointChangeType.CONSUME.getCode(), 0); | 
|---|
|  |  |  | int sharePoint = userBalanceMap.getOrDefault(PointChangeType.REBATE.getCode(), 0); | 
|---|
|  |  |  | int pullNewPoint = userBalanceMap.getOrDefault(PointChangeType.PULL_NEW.getCode(), 0); | 
|---|
|  |  |  | int registerPoint = userBalanceMap.getOrDefault(PointChangeType.REGISTER.getCode(), 0); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 做工积分:签到积分 + 每日分享积分 + 每日签到积分 + 每日使用时长积分 | 
|---|
|  |  |  | int share = userBalanceMap.getOrDefault(PointChangeType.SHARE.getCode(), 0); | 
|---|
|  |  |  | int signIn = userBalanceMap.getOrDefault(PointChangeType.SIGN_IN.getCode(), 0); | 
|---|
|  |  |  | int useTime = userBalanceMap.getOrDefault(PointChangeType.USE_TIME.getCode(), 0); | 
|---|
|  |  |  | int workPoint = share + signIn + useTime; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int shopAchievementPoint = userBalanceMap.getOrDefault(PointChangeType.TECHNICIAN_ACHIEVEMENT.getCode(), 0); | 
|---|
|  |  |  | //总积分 | 
|---|
|  |  |  | int totalPoint = consumePoint+sharePoint+pullNewPoint+registerPoint+workPoint+shopAchievementPoint; | 
|---|
|  |  |  | UserPointStatistics userPointStatistics = new UserPointStatistics(); | 
|---|
|  |  |  | userPointStatistics.setTotalPoint(totalPoint); | 
|---|
|  |  |  | userPointStatistics.setConsumePoint(consumePoint); | 
|---|
|  |  |  | userPointStatistics.setSharePoint(sharePoint); | 
|---|
|  |  |  | userPointStatistics.setPullNewPoint(pullNewPoint); | 
|---|
|  |  |  | userPointStatistics.setRegisterPoint(registerPoint); | 
|---|
|  |  |  | userPointStatistics.setWorkPoint(workPoint); | 
|---|
|  |  |  | userPointStatistics.setShopAchievementPoint(shopAchievementPoint); | 
|---|
|  |  |  | return userPointStatistics; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Override | 
|---|
|  |  |  | public IPage<UserPoint> getUserPointPage(Page<UserPoint> page, UserPoint userPoint) { | 
|---|
|  |  |  | return this.baseMapper.queryUserPointPage(page, userPoint); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|