热门产品销售看板

This commit is contained in:
2025-07-26 10:18:16 +08:00
parent a78bf64529
commit bb42f9c5b7
11 changed files with 363 additions and 4 deletions

View File

@@ -22,6 +22,9 @@ import com.klp.domain.vo.WmsProductSalesScriptVo;
import com.klp.domain.bo.WmsProductSalesScriptBo;
import com.klp.service.IWmsProductSalesScriptService;
import com.klp.common.core.page.TableDataInfo;
import com.klp.common.utils.redis.RedisUtils;
import java.time.Duration;
import com.klp.domain.vo.HotProductVO;
/**
* 产品销售话术
@@ -97,4 +100,29 @@ public class WmsProductSalesScriptController extends BaseController {
@PathVariable Long[] scriptIds) {
return toAjax(iWmsProductSalesScriptService.deleteWithValidByIds(Arrays.asList(scriptIds), true));
}
/**
* 记录话术访问频率
*
* @param productId 产品ID
*/
@PostMapping("/recordVisit/{productId}")
public R<Void> recordVisit(@NotNull(message = "产品ID不能为空") @PathVariable Long productId) {
String key = "product:visit:frequency:" + productId;
// 原子递增访问次数设置7天过期时间
long count = RedisUtils.incrAtomicValue(key);
if (count == 1) {
// 第一次访问时设置过期时间
RedisUtils.expire(key, Duration.ofDays(7));
}
return R.ok();
}
/**
* 获取热门产品排行(基于访问频率)
*/
@GetMapping("/hotProducts")
public R<List<HotProductVO>> getHotProducts(@RequestParam(defaultValue = "10") Integer limit) {
return R.ok(iWmsProductSalesScriptService.getHotProducts(limit));
}
}

View File

@@ -155,9 +155,6 @@ public class WmsStockIoController extends BaseController {
* 扫码枪专用根据明细ID直接入库无需审核整单
*/
@PostMapping("/scanInStock")
/**
* 扫码枪专用:根据传入明细参数直接入库,无需审核整单
*/
public R<Void> scanInStock(@RequestBody WmsStockIoDetailBo bo) {
try {
boolean result = iWmsStockIoService.scanInStockByBo(bo);

View File

@@ -3,11 +3,42 @@ package com.klp.domain.vo;
import lombok.Data;
import java.util.List;
/**
* 数据看板概览视图对象
*
* @author klp
* @date 2025-01-27
*/
@Data
public class DashboardOverviewVO {
/**
* 订单汇总
*/
private OrderSummaryVO orderSummary;
/**
* 销售经理饼图
*/
private List<SalesManagerPieVO> salesManagerPie;
/**
* 产品销量排行
*/
private List<ProductRankVO> productRank;
/**
* 订单物料分析
*/
private List<OrderMaterialVO> orderMaterial;
/**
* 客户分布
*/
private List<CustomerRegionVO> customerRegion;
/**
* 热门产品排行(基于访问频率)
*/
private List<HotProductVO> hotProducts;
}

View File

@@ -0,0 +1,38 @@
package com.klp.domain.vo;
import lombok.Data;
/**
* 热门产品视图对象
*
* @author klp
* @date 2025-01-27
*/
@Data
public class HotProductVO {
/**
* 产品ID
*/
private Long productId;
/**
* 产品名称
*/
private String productName;
/**
* 产品编号
*/
private String productCode;
/**
* 访问次数
*/
private Long visitCount;
/**
* 排名
*/
private Integer rank;
}

View File

@@ -5,6 +5,7 @@ import com.klp.domain.vo.WmsProductSalesScriptVo;
import com.klp.domain.bo.WmsProductSalesScriptBo;
import com.klp.common.core.page.TableDataInfo;
import com.klp.common.core.domain.PageQuery;
import com.klp.domain.vo.HotProductVO;
import java.util.Collection;
import java.util.List;
@@ -47,4 +48,8 @@ public interface IWmsProductSalesScriptService {
*/
Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid);
/**
* 获取热门产品排行(基于访问频率)
*/
List<HotProductVO> getHotProducts(Integer limit);
}

View File

@@ -14,6 +14,12 @@ import com.klp.domain.vo.WmsProductSalesScriptVo;
import com.klp.domain.WmsProductSalesScript;
import com.klp.mapper.WmsProductSalesScriptMapper;
import com.klp.service.IWmsProductSalesScriptService;
import com.klp.domain.vo.HotProductVO;
import com.klp.common.utils.redis.RedisUtils;
import java.util.stream.Collectors;
import java.util.Comparator;
import com.klp.domain.WmsProduct;
import com.klp.mapper.WmsProductMapper;
import java.util.List;
import java.util.Map;
@@ -30,6 +36,7 @@ import java.util.Collection;
public class WmsProductSalesScriptServiceImpl implements IWmsProductSalesScriptService {
private final WmsProductSalesScriptMapper baseMapper;
private final WmsProductMapper wmsProductMapper;
/**
* 查询产品销售话术
@@ -117,4 +124,44 @@ public class WmsProductSalesScriptServiceImpl implements IWmsProductSalesScriptS
}
return baseMapper.deleteBatchIds(ids) > 0;
}
/**
* 获取热门产品排行(基于访问频率)
*/
@Override
public List<HotProductVO> getHotProducts(Integer limit) {
// 获取所有产品访问频率的key
Collection<String> keys = RedisUtils.keys("product:visit:frequency:*");
List<HotProductVO> hotProducts = keys.stream()
.map(key -> {
String productIdStr = key.replace("product:visit:frequency:", "");
Long productId = Long.valueOf(productIdStr);
Long visitCount = RedisUtils.getAtomicValue(key);
// 查询产品信息
WmsProduct product = wmsProductMapper.selectById(productId);
if (product == null) {
return null;
}
HotProductVO vo = new HotProductVO();
vo.setProductId(productId);
vo.setProductName(product.getProductName());
vo.setProductCode(product.getProductCode());
vo.setVisitCount(visitCount);
return vo;
})
.filter(vo -> vo != null)
.sorted(Comparator.comparing(HotProductVO::getVisitCount).reversed())
.limit(limit)
.collect(Collectors.toList());
// 设置排名
for (int i = 0; i < hotProducts.size(); i++) {
hotProducts.get(i).setRank(i + 1);
}
return hotProducts;
}
}

View File

@@ -16,6 +16,8 @@ import com.klp.domain.WmsProduct;
import com.klp.mapper.WmsProductMapper;
import com.klp.service.IWmsProductService;
import com.klp.domain.vo.OrderSummaryVO;
import com.klp.domain.vo.HotProductVO;
import com.klp.service.IWmsProductSalesScriptService;
import java.util.List;
import java.util.Map;
@@ -32,6 +34,7 @@ import java.util.Collection;
public class WmsProductServiceImpl implements IWmsProductService {
private final WmsProductMapper baseMapper;
private final IWmsProductSalesScriptService iWmsProductSalesScriptService;
/**
* 查询产品
@@ -149,6 +152,8 @@ public class WmsProductServiceImpl implements IWmsProductService {
vo.setProductRank(baseMapper.selectProductRank());
vo.setOrderMaterial(baseMapper.selectOrderMaterial());
vo.setCustomerRegion(baseMapper.selectCustomerRegion());
// 添加热门产品数据
vo.setHotProducts(iWmsProductSalesScriptService.getHotProducts(10));
return vo;
}
}