package cn.lili.modules.member.serviceimpl; import cn.lili.base.Result; import cn.lili.common.security.context.UserContext; import cn.lili.common.utils.StringUtils; import cn.lili.modules.goods.entity.dos.GoodsSku; import cn.lili.modules.goods.service.GoodsSkuService; import cn.lili.modules.lmk.domain.query.FootPrintQuery; import cn.lili.modules.lmk.domain.vo.*; import cn.lili.modules.lmk.mapper.VideoMapper; import cn.lili.modules.member.entity.dos.FootPrint; import cn.lili.modules.member.entity.dto.FootPrintQueryParams; import cn.lili.modules.member.mapper.FootprintMapper; import cn.lili.modules.member.service.FootprintService; import cn.lili.modules.search.entity.dos.EsGoodsIndex; import cn.lili.modules.statistics.entity.dto.StatisticsQueryParam; import cn.lili.mybatis.util.PageUtil; import cn.lili.utils.COSUtil; import cn.lili.utils.CommonUtil; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import java.math.BigDecimal; import java.math.RoundingMode; import java.text.SimpleDateFormat; import java.util.*; import java.util.stream.Collectors; import java.util.stream.IntStream; /** * 会员浏览历史业务层实现 * * @author Chopper * @since 2020/11/18 10:46 上午 */ @Service public class FootprintServiceImpl extends ServiceImpl implements FootprintService { @Autowired private GoodsSkuService goodsSkuService; @Autowired private VideoMapper videoMapper; @Autowired private COSUtil cOSUtil; @Override public FootPrint saveFootprint(FootPrint footPrint) { this.save(footPrint); return footPrint; } @Override public boolean clean() { LambdaQueryWrapper lambdaQueryWrapper = Wrappers.lambdaQuery(); lambdaQueryWrapper.eq(FootPrint::getMemberId, UserContext.getCurrentUser().getId()); return this.remove(lambdaQueryWrapper); } @Override public boolean deleteByIds(List ids, String viewType) { LambdaQueryWrapper lambdaQueryWrapper = Wrappers.lambdaQuery(); lambdaQueryWrapper.eq(FootPrint::getMemberId, UserContext.getCurrentUser().getId()); lambdaQueryWrapper.eq(FootPrint::getViewType, viewType); lambdaQueryWrapper.in(FootPrint::getRefId, ids); return this.remove(lambdaQueryWrapper); } @Override public IPage footPrintPage(FootPrintQueryParams params) { params.setSort("createTime"); Page footPrintPages = this.page(PageUtil.initPage(params), params.queryWrapper()); //定义结果 Page esGoodsIndexIPage = new Page<>(); if (footPrintPages.getRecords() != null && !footPrintPages.getRecords().isEmpty()) { List skuIds = footPrintPages.getRecords().stream().map(FootPrint::getSkuId).collect(Collectors.toList()); List goodsSkuByIdFromCache = goodsSkuService.getGoodsSkuByIdFromCache(skuIds); List collect = IntStream.range(0, goodsSkuByIdFromCache.size()) .mapToObj(i -> { if (goodsSkuByIdFromCache.get(i) == null) { return null; } Optional first = footPrintPages.getRecords().stream().filter(j -> goodsSkuByIdFromCache.get(i).getId().equals(j.getSkuId())).findFirst(); return first.map(footPrint -> new EsGoodsIndex(goodsSkuByIdFromCache.get(i), footPrint.getCreateTime())).orElseGet(() -> new EsGoodsIndex(goodsSkuByIdFromCache.get(i))); }) .collect(Collectors.toList()); collect.removeIf(Objects::isNull); esGoodsIndexIPage.setPages(footPrintPages.getPages()); esGoodsIndexIPage.setRecords(collect); esGoodsIndexIPage.setTotal(footPrintPages.getTotal()); esGoodsIndexIPage.setSize(footPrintPages.getSize()); esGoodsIndexIPage.setCurrent(footPrintPages.getCurrent()); return esGoodsIndexIPage; } return esGoodsIndexIPage; } @Override public Result videoFootPrintPage(FootPrintQuery query) { IPage page = cn.lili.utils.PageUtil.getPage(query,VideoFootVO.class); videoMapper.videoFootPage(page,query); VideoFootInfoVo videoFootInfoVo = videoMapper.getVideoFootInfo(query.getMemberId()); for (VideoFootVO videoFootVO : page.getRecords()) { videoFootVO.setCoverCOSUrl(cOSUtil.getPreviewUrl(videoFootVO.getCoverUrl())); //计算播放进度 if(videoFootVO.getPlayAt() == null || videoFootVO.getPlayAt().isEmpty()){ videoFootVO.setPlayProgress(0.0); continue; } BigDecimal value = BigDecimal.valueOf(Double.parseDouble(videoFootVO.getPlayAt()) / videoFootVO.getVideoDuration()) .setScale(2, RoundingMode.HALF_UP); if (value.compareTo(BigDecimal.ONE) > 0) { // 这里可以添加处理逻辑,比如:自动修正为1.0 value = BigDecimal.ONE; } videoFootVO.setPlayProgress(value.doubleValue()); } HashMap map = new HashMap<>(); map.put("data", page.getRecords()); map.put("total", page.getTotal()); if (videoFootInfoVo == null) { map.put("avgProgress",0); map.put("totalDuration",0); }else { map.put("avgProgress",videoFootInfoVo.getAvgCompletionRate()); map.put("totalDuration",videoFootInfoVo.getTotalDuration()); } return Result.ok().data(map); } @Override public Result memberActionAnalyse(String id){ //视频分类 List list = videoMapper.videoFoot(id); Map tagCountMap = list.stream() .flatMap(video -> video.getTagList().stream()) // 展开所有 tag .filter(Objects::nonNull) // 过滤 null .collect(Collectors.groupingBy( SimpleVideoTagVO::getTagName, // 按 tagName 分组 Collectors.counting() // 计算每个 tagName 出现的次数 )); return Result.ok().data(tagCountMap); } @Override public long getFootprintNum() { LambdaQueryWrapper lambdaQueryWrapper = Wrappers.lambdaQuery(); lambdaQueryWrapper.eq(FootPrint::getMemberId, Objects.requireNonNull(UserContext.getCurrentUser()).getId()); lambdaQueryWrapper.eq(FootPrint::getDeleteFlag, false); return this.count(lambdaQueryWrapper); } @Override public Result getViewAndCompletionRateCount(StatisticsQueryParam queryParam) { Date startTime = null; Date endTime = new Date(); // 结束时间默认是当前时间 List dateList = new ArrayList<>(); Calendar calendar = Calendar.getInstance(); // 用于日期计算的日历实例 SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); int days = 0; if (queryParam.getYear() != null && queryParam.getMonth() != null) { Date[] dates = CommonUtil.getMonthStartAndEnd(queryParam.getYear(),queryParam.getMonth()); startTime = dates[0]; endTime = dates[1]; }else { switch (queryParam.getSearchType()) { case "TODAY": // 今天:从今天0点到现在 calendar.setTime(new Date()); // 重置为当前时间 calendar.set(Calendar.HOUR_OF_DAY, 0); // 小时设为0(24小时制) calendar.set(Calendar.MINUTE, 0); // 分钟设为0 calendar.set(Calendar.SECOND, 0); // 秒设为0 calendar.set(Calendar.MILLISECOND, 0); // 毫秒设为0 startTime = calendar.getTime(); // 得到今天0点的Date对象 dateList.add(sdf.format(startTime)); break; case "YESTERDAY": // 昨天:从昨天0点到昨天23:59:59.999 calendar.setTime(new Date()); calendar.add(Calendar.DATE, -1); // 日期减1天(变为昨天) // 设置昨天0点 calendar.set(Calendar.HOUR_OF_DAY, 0); calendar.set(Calendar.MINUTE, 0); calendar.set(Calendar.SECOND, 0); calendar.set(Calendar.MILLISECOND, 0); startTime = calendar.getTime(); // 设置昨天23:59:59.999 calendar.set(Calendar.HOUR_OF_DAY, 23); calendar.set(Calendar.MINUTE, 59); calendar.set(Calendar.SECOND, 59); calendar.set(Calendar.MILLISECOND, 999); endTime = calendar.getTime(); dateList.add(sdf.format(startTime)); break; case "LAST_SEVEN": // 过去七天:从7天前0点到现在(含今天共7天) calendar.setTime(new Date()); calendar.add(Calendar.DATE, -6); // 日期减6天(7天前的今天) // 设置7天前0点 calendar.set(Calendar.HOUR_OF_DAY, 0); calendar.set(Calendar.MINUTE, 0); calendar.set(Calendar.SECOND, 0); calendar.set(Calendar.MILLISECOND, 0); startTime = calendar.getTime(); days = 7; // 循环生成7天的日期字符串 Calendar tempCalendar = Calendar.getInstance(); tempCalendar.setTime(startTime); for (int i = 0; i < days; i++) { dateList.add(sdf.format(tempCalendar.getTime())); tempCalendar.add(Calendar.DATE, 1); // 每天累加1天 } break; case "LAST_THIRTY": // 过去30天:从30天前0点到现在(含今天共30天) calendar.setTime(new Date()); calendar.add(Calendar.DATE, -29); // 日期减29天(30天前的今天) // 设置30天前0点 calendar.set(Calendar.HOUR_OF_DAY, 0); calendar.set(Calendar.MINUTE, 0); calendar.set(Calendar.SECOND, 0); calendar.set(Calendar.MILLISECOND, 0); startTime = calendar.getTime(); days = 30; // 循环生成30天的日期字符串 Calendar tempCalendar30 = Calendar.getInstance(); tempCalendar30.setTime(startTime); for (int i = 0; i < days; i++) { dateList.add(sdf.format(tempCalendar30.getTime())); tempCalendar30.add(Calendar.DATE, 1); // 每天累加1天 } break; default: return Result.error("不支持的时间范围类型"); } } if ("goods".equals(queryParam.getCurrentType())){ List> maps = baseMapper.selectViewAndCompletionRateCountByDay(startTime, endTime,queryParam.getCurrentLimit()); Map viewPrintMap = maps.stream() .collect(Collectors.toMap( map -> { String goodsId = Objects.toString(map.get("GoodsId"), ""); // 空值处理 String goodsName = Objects.toString(map.get("GoodsName"), ""); // 空值处理 return goodsName + " | " + goodsId; // 用|分隔,方便后续拆分 }, map-> Long.valueOf(map.get("view_count").toString()), (existing, replacement) -> existing, LinkedHashMap::new // 指定使用 LinkedHashMap,保留插入顺序 )); List yData = new ArrayList<>(); List xData = new ArrayList<>(); for (Map.Entry entry : viewPrintMap.entrySet()){ yData.add(entry.getKey()); xData.add(entry.getValue()); } Map data = new HashMap<>(); data.put("yData", yData); data.put("xData", xData); return Result.ok().data(data); }else{ List> maps = baseMapper.selectEachVideoStats(startTime, endTime,queryParam.getCurrentLimit()); Map viewPrintMap = maps.stream() .collect(Collectors.toMap( map -> { String videoId = Objects.toString(map.get("video_id"), ""); // 空值处理 String videoName = Objects.toString(map.get("title"), ""); // 空值处理 return videoName + " | " + videoId; // 用|分隔,方便后续拆分 }, map-> { VideoViewCompletionRateVO vo = new VideoViewCompletionRateVO(); vo.setTotal_views(Long.valueOf(map.get("total_views").toString())); vo.setComplete_rate(map.get("complete_rate").toString()); return vo; }, (existing, replacement) -> existing, LinkedHashMap::new // 指定使用 LinkedHashMap,保留插入顺序 )); List yData = new ArrayList<>(); List xData = new ArrayList<>(); List rateData = new ArrayList<>(); for (Map.Entry entry : viewPrintMap.entrySet()){ yData.add(entry.getKey()); xData.add(entry.getValue().getTotal_views()); rateData.add(entry.getValue().getComplete_rate()); } Map data = new HashMap<>(); data.put("yData", yData); data.put("xData", xData); data.put("rateData", rateData); return Result.ok().data(data); } } }