商城:完善购物车的逻辑
parent
2113e825d7
commit
da162853ec
@ -0,0 +1,45 @@
|
||||
package cn.iocoder.yudao.module.trade.controller.app.cart.vo;
|
||||
|
||||
import cn.iocoder.yudao.module.trade.controller.app.base.sku.AppProductSkuBaseRespVO;
|
||||
import cn.iocoder.yudao.module.trade.controller.app.base.spu.AppProductSpuBaseRespVO;
|
||||
import io.swagger.v3.oas.annotations.media.Schema;
|
||||
import lombok.Data;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
@Schema(description = "用户 App - 用户的购物列表 Response VO")
|
||||
@Data
|
||||
public class AppTradeCartListRespVO {
|
||||
|
||||
/**
|
||||
* 有效的购物项数组
|
||||
*/
|
||||
private List<Cart> validList;
|
||||
|
||||
/**
|
||||
* 无效的购物项数组
|
||||
*/
|
||||
private List<Cart> invalidList;
|
||||
|
||||
@Schema(description = "购物项")
|
||||
@Data
|
||||
public static class Cart {
|
||||
|
||||
@Schema(description = "购物项的编号", required = true, example = "1024")
|
||||
private Long id;
|
||||
|
||||
@Schema(description = "商品数量", required = true, example = "1")
|
||||
private Integer count;
|
||||
|
||||
/**
|
||||
* 商品 SPU
|
||||
*/
|
||||
private AppProductSpuBaseRespVO spu;
|
||||
/**
|
||||
* 商品 SKU
|
||||
*/
|
||||
private AppProductSkuBaseRespVO sku;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
@ -1,45 +1,52 @@
|
||||
package cn.iocoder.yudao.module.trade.convert.cart;
|
||||
|
||||
import cn.iocoder.yudao.module.promotion.api.price.dto.PriceCalculateReqDTO;
|
||||
import cn.iocoder.yudao.module.promotion.api.price.dto.PriceCalculateRespDTO;
|
||||
import cn.iocoder.yudao.module.trade.controller.app.cart.vo.AppTradeCartDetailRespVO;
|
||||
import cn.iocoder.yudao.module.trade.dal.dataobject.cart.TradeCartItemDO;
|
||||
import cn.iocoder.yudao.module.product.api.sku.dto.ProductSkuRespDTO;
|
||||
import cn.iocoder.yudao.module.product.api.spu.dto.ProductSpuRespDTO;
|
||||
import cn.iocoder.yudao.module.product.enums.spu.ProductSpuStatusEnum;
|
||||
import cn.iocoder.yudao.module.trade.controller.app.base.sku.AppProductSkuBaseRespVO;
|
||||
import cn.iocoder.yudao.module.trade.controller.app.base.spu.AppProductSpuBaseRespVO;
|
||||
import cn.iocoder.yudao.module.trade.controller.app.cart.vo.AppTradeCartListRespVO;
|
||||
import cn.iocoder.yudao.module.trade.dal.dataobject.cart.TradeCartDO;
|
||||
import org.mapstruct.Mapper;
|
||||
import org.mapstruct.Mapping;
|
||||
import org.mapstruct.Mappings;
|
||||
import org.mapstruct.factory.Mappers;
|
||||
|
||||
import java.util.Collections;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertList;
|
||||
import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertMap;
|
||||
|
||||
@Mapper
|
||||
public interface TradeCartConvert {
|
||||
|
||||
TradeCartConvert INSTANCE = Mappers.getMapper(TradeCartConvert.class);
|
||||
|
||||
default AppTradeCartDetailRespVO buildEmptyAppTradeCartDetailRespVO() {
|
||||
return new AppTradeCartDetailRespVO().setItemGroups(Collections.emptyList())
|
||||
.setOrder(new AppTradeCartDetailRespVO.Order().setSkuOriginalPrice(0).setSkuPromotionPrice(0)
|
||||
.setOrderPromotionPrice(0).setDeliveryPrice(0).setPayPrice(0));
|
||||
default AppTradeCartListRespVO convertList(List<TradeCartDO> carts,
|
||||
List<ProductSpuRespDTO> spus, List<ProductSkuRespDTO> skus) {
|
||||
Map<Long, ProductSpuRespDTO> spuMap = convertMap(spus, ProductSpuRespDTO::getId);
|
||||
Map<Long, ProductSkuRespDTO> skuMap = convertMap(skus, ProductSkuRespDTO::getId);
|
||||
// 遍历,开始转换
|
||||
List<AppTradeCartListRespVO.Cart> validList = new ArrayList<>(carts.size());
|
||||
List<AppTradeCartListRespVO.Cart> invalidList = new ArrayList<>();
|
||||
carts.forEach(cart -> {
|
||||
AppTradeCartListRespVO.Cart cartVO = new AppTradeCartListRespVO.Cart();
|
||||
cartVO.setId(cart.getId()).setCount(cart.getCount());
|
||||
ProductSpuRespDTO spu = spuMap.get(cart.getSpuId());
|
||||
ProductSkuRespDTO sku = skuMap.get(cart.getSkuId());
|
||||
cartVO.setSpu(convert(spu)).setSku(convert(sku));
|
||||
// 如果 spu 或 sku 不存在,或者 spu 被禁用,说明是非法的,或者 sku 库存不足
|
||||
if (spu == null
|
||||
|| sku == null
|
||||
|| !ProductSpuStatusEnum.isEnable(spu.getStatus())
|
||||
|| sku.getStock() <= 0) {
|
||||
invalidList.add(cartVO);
|
||||
} else {
|
||||
validList.add(cartVO);
|
||||
}
|
||||
});
|
||||
return new AppTradeCartListRespVO().setValidList(validList).setValidList(invalidList);
|
||||
}
|
||||
|
||||
default PriceCalculateReqDTO convert(Long userId, List<TradeCartItemDO> cartItems) {
|
||||
return new PriceCalculateReqDTO().setUserId(userId)
|
||||
.setItems(convertList(cartItems, cartItem -> new PriceCalculateReqDTO.Item().setSkuId(cartItem.getSkuId())
|
||||
.setCount(cartItem.getSelected() ? cartItem.getCount() : 0)));
|
||||
}
|
||||
|
||||
// ========== AppTradeCartDetailRespVO 相关 ==========
|
||||
|
||||
AppTradeCartDetailRespVO.Promotion convert(PriceCalculateRespDTO.Promotion bean);
|
||||
|
||||
@Mappings({
|
||||
@Mapping(source = "cartItem.count", target = "count")
|
||||
})
|
||||
AppTradeCartDetailRespVO.Sku convert(PriceCalculateRespDTO.OrderItem orderItem, TradeCartItemDO cartItem);
|
||||
|
||||
AppTradeCartDetailRespVO.Order convert(PriceCalculateRespDTO.Order bean);
|
||||
AppProductSpuBaseRespVO convert(ProductSpuRespDTO spu);
|
||||
AppProductSkuBaseRespVO convert(ProductSkuRespDTO sku);
|
||||
|
||||
}
|
||||
|
||||
@ -1,47 +0,0 @@
|
||||
package cn.iocoder.yudao.module.trade.dal.mysql.cart;
|
||||
|
||||
import cn.hutool.core.collection.CollUtil;
|
||||
import cn.hutool.core.map.MapUtil;
|
||||
import cn.iocoder.yudao.framework.mybatis.core.mapper.BaseMapperX;
|
||||
import cn.iocoder.yudao.framework.mybatis.core.query.LambdaQueryWrapperX;
|
||||
import cn.iocoder.yudao.module.trade.dal.dataobject.cart.TradeCartItemDO;
|
||||
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
||||
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
|
||||
import org.apache.ibatis.annotations.Mapper;
|
||||
|
||||
import java.util.Collection;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
@Mapper
|
||||
public interface TradeCartItemMapper extends BaseMapperX<TradeCartItemDO> {
|
||||
|
||||
default TradeCartItemDO selectByUserIdAndSkuId(Long userId, Long skuId) {
|
||||
return selectOne(TradeCartItemDO::getUserId, userId,
|
||||
TradeCartItemDO::getSkuId, skuId);
|
||||
}
|
||||
|
||||
default List<TradeCartItemDO> selectListByUserIdAndSkuIds(Long userId, Collection<Long> skuIds) {
|
||||
return selectList(new LambdaQueryWrapper<TradeCartItemDO>().eq(TradeCartItemDO::getUserId, userId)
|
||||
.in(TradeCartItemDO::getSkuId, skuIds));
|
||||
}
|
||||
|
||||
default void updateByIds(Collection<Long> ids, TradeCartItemDO updateObject) {
|
||||
update(updateObject, new LambdaQueryWrapper<TradeCartItemDO>().in(TradeCartItemDO::getId, ids));
|
||||
}
|
||||
|
||||
default Integer selectSumByUserId(Long userId) {
|
||||
// SQL sum 查询
|
||||
List<Map<String, Object>> result = selectMaps(new QueryWrapper<TradeCartItemDO>()
|
||||
.select("SUM(count) AS sumCount")
|
||||
.eq("user_id", userId));
|
||||
// 获得数量
|
||||
return CollUtil.isNotEmpty(result) ? MapUtil.getInt(result.get(0), "sumCount") : 0;
|
||||
}
|
||||
|
||||
default List<TradeCartItemDO> selectListByUserId(Long userId, Boolean selected) {
|
||||
return selectList(new LambdaQueryWrapperX<TradeCartItemDO>().eq(TradeCartItemDO::getUserId, userId)
|
||||
.eqIfPresent(TradeCartItemDO::getSelected, selected));
|
||||
}
|
||||
|
||||
}
|
||||
@ -0,0 +1,56 @@
|
||||
package cn.iocoder.yudao.module.trade.dal.mysql.cart;
|
||||
|
||||
import cn.hutool.core.collection.CollUtil;
|
||||
import cn.hutool.core.map.MapUtil;
|
||||
import cn.iocoder.yudao.framework.mybatis.core.mapper.BaseMapperX;
|
||||
import cn.iocoder.yudao.module.trade.dal.dataobject.cart.TradeCartDO;
|
||||
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
||||
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
|
||||
import org.apache.ibatis.annotations.Mapper;
|
||||
|
||||
import java.util.Collection;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
@Mapper
|
||||
public interface TradeCartMapper extends BaseMapperX<TradeCartDO> {
|
||||
|
||||
default TradeCartDO selectByUserIdAndSkuId(Long userId, Long skuId,
|
||||
Boolean addStatus, Boolean orderStatus) {
|
||||
return selectOne(TradeCartDO::getUserId, userId,
|
||||
TradeCartDO::getSkuId, skuId,
|
||||
TradeCartDO::getAddStatus, addStatus,
|
||||
TradeCartDO::getOrderStatus, orderStatus);
|
||||
}
|
||||
|
||||
default Integer selectSumByUserId(Long userId) {
|
||||
// SQL sum 查询
|
||||
List<Map<String, Object>> result = selectMaps(new QueryWrapper<TradeCartDO>()
|
||||
.select("SUM(count) AS sumCount")
|
||||
.eq("user_id", userId)
|
||||
.eq("add_status", true) // 只计算添加到购物车中的
|
||||
.eq("order_status", false)); // 必须未下单
|
||||
// 获得数量
|
||||
return CollUtil.getFirst(result) != null ? MapUtil.getInt(result.get(0), "sumCount") : 0;
|
||||
}
|
||||
|
||||
default TradeCartDO selectById(Long id, Long userId) {
|
||||
return selectOne(TradeCartDO::getId, id,
|
||||
TradeCartDO::getUserId, userId);
|
||||
}
|
||||
|
||||
default List<TradeCartDO> selectListByIds(Collection<Long> ids, Long userId) {
|
||||
return selectList(new LambdaQueryWrapper<TradeCartDO>()
|
||||
.in(TradeCartDO::getId, ids)
|
||||
.eq(TradeCartDO::getUserId, userId));
|
||||
}
|
||||
|
||||
default List<TradeCartDO> selectListByUserId(Long userId) {
|
||||
return selectList(TradeCartDO::getUserId, userId);
|
||||
}
|
||||
|
||||
default void updateByIds(Collection<Long> ids, TradeCartDO updateObject) {
|
||||
update(updateObject, new LambdaQueryWrapper<TradeCartDO>().in(TradeCartDO::getId, ids));
|
||||
}
|
||||
|
||||
}
|
||||
Loading…
Reference in New Issue