外卖项目06---套餐管理业务开发(移动端的后台代码编辑开发)

菜品展示、购物车、下单

目录

一、导入用户地址簿相关功能代码 90

1.1需求分析 90

1.2数据模型 90

1.3导入功能代码 90

二、菜品展示 91

2.1需求分析 91

2.2商品展示---代码开发---梳理交互过程 92

2.3菜品展示---代码开发---修改DishController的list方法并测试 93

2.4菜品展示---代码开发---创建SetmealController的list方法并测试 94

三、购物车 95

3.1需求分析 95

3.2数据模型 95

3.3代码开发---梳理交互过程 95

3.4代码开发---准备工作 95

3.5购物车---代码开发---添加购物车 96 

3.6购物车---代码开发---查看购物车&清空购物车 97

四、用户下单 98

4.1需求分析 98

4.2数据模型 98

4.3用户下单---代码开发---梳理交互过程&准备工作 99

4.4用户下单---准备工作 99 

4.5用户下单---代码开发 100 

4.6用户下单---代码开发(在OrderServiceImpl驱动类实现层实现submit方法实现)101

4.7用户下的那---代码开发(向订单表中进行数据的单条插入,订单明细多条数据插入,清空购物车数据)102


一、导入用户地址簿相关功能代码 90

1.1需求分析 90

1.2数据模型 90

1.3导入功能代码 90

二、菜品展示 91

2.1需求分析 91

2.2商品展示---代码开发---梳理交互过程 92

两个需求分析: 

在客户端请求数据的展示:

前端部分的功能展示:和上面的两次list对应的关系 

前端设置的是,这两部分同时正确的前提下,才能够在前端页面渲染成功。

目前的情况是:请求购物车的加载失败,所以这条渲染才没能够成功。

修改购物车请求的内容:

进行修改后:如果界面没有显示,可以使用ctrl+f5,进行刷新浏览器的缓存

2.3菜品展示---代码开发---修改DishController的list方法并测试 93

本部分功能主要实现的是,在移动端的界面能够进行点击“选择规格”后,会弹出口味的信息

本部分代码的实现,采用的就是之前展示菜品还要展示菜品对应的名称,这种类型的代码形式。

 本部分代码实现:

    /**
     * 根据条件查询对应的菜品数据
     * @param dish
     * @return
     */
    @GetMapping("/list")
    public R<List<DishDto>> list(Dish dish){//这个部分传递的参数,可以是Long categoryid,
                                        // 但是可以将其进行扩展使用,使用Dish类中的参数,它包含了很多的参数信息,包含categoryid
        //构造查询条件
        LambdaQueryWrapper<Dish> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(dish.getCategoryId() != null , Dish::getCategoryId,dish.getCategoryId());
                                                            //该部分是设置一个判断语句,确定id传递过来,并不是空的指针
        //添加排序条件
        queryWrapper.orderByAsc(Dish::getSort).orderByDesc(Dish::getUpdateTime);
                                        //使用了两个判断条件,一个是根据sort进行升序排序,如果sort排序一致,再根据更新时间进行降序判断
        //添加条件,查询状态为1(起售状态)的菜品
        queryWrapper.eq(Dish::getStatus,1);
        List<Dish> list = dishService.list(queryWrapper);//使用该方法,获取得到了一个集合

        List<DishDto> dishDtoList = list.stream().map((item)->{
            DishDto dishDto = new DishDto();
            BeanUtils.copyProperties(item,dishDto);

            Long categoryId = item.getCategoryId();//分类id
            //根据id查询分类对象
            Category category = categoryService.getById(categoryId);

            if(category != null){
                String categoryName = category.getName();
                dishDto.setCategoryName(categoryName);
            }
            //当前菜品的id
            Long dishId = item.getId();
            //实现SQL语句:select * from dish_flavor where dish_id = ?
            LambdaQueryWrapper<DishFlavor> lambdaQueryWrapper = new LambdaQueryWrapper<>();
            lambdaQueryWrapper.eq(DishFlavor::getDishId,dishId);

            List<DishFlavor> dishFlavorList = dishFlavorService.list(lambdaQueryWrapper);
            dishDto.setFlavors(dishFlavorList);//将上面查询得到的结果设置到dishDto中
            return dishDto;//将处理后的数据结果返回出去
        }).collect(Collectors.toList());
        return R.success(dishDtoList);
    }

2.4菜品展示---代码开发---创建SetmealController的list方法并测试 94

本部分需要解决的问题:套餐点击后出现接口异常。

        对应的请求路径信息如下所示:但是之前程序代码编写的过程中并没有该该部分,所以需要进行编写该部分的代码。需要在setmealController中创建list方法。

        本部分实际上就是通过IDEA编写代码,能够从数据库内根据自己需要的条件信息,来判断,获取得到对应的数据信息,进行反馈给移动端。

本部分实现的效果:

 本部分的代码:在setmealController中添加list方法

    /**
     *根据条件查询套餐数据
     * @param setmeal
     * @return
     */
    @GetMapping("/list")
    public R<List<Setmeal>> list(Setmeal setmeal){
        //SQL 筛选判断,两个等值判断+一个根据更新时间顺序进行降序排列
        LambdaQueryWrapper<Setmeal> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(setmeal.getCategoryId() != null,Setmeal::getCategoryId,setmeal.getCategoryId());
        queryWrapper.eq(setmeal.getStatus() != null,Setmeal::getStatus,setmeal.getStatus());

        List<Setmeal> list = setmealService.list(queryWrapper);


        return R.success(list);
    }

三、购物车 95

3.1需求分析 95

将菜品或者菜品套餐添加到购物车,都是使用的同一个模型 

注:一共有三个需求进行填写 

3.2数据模型 95

3.3代码开发---梳理交互过程 95

3.4代码开发---准备工作 95

3.5购物车---代码开发---添加购物车 96 

效果实现:实现商品添加购物车添加成功的功能

 发送请求时,会有如下的操作指示:

实现效果:

1、菜品能够直接添加,并且能够在购物车内进行数量的增加

2、套餐的添加

本部分代码如下所示:

    /**
     * 添加购物车
     * @param shoppingCart
     * @return
     */
    @PostMapping("/add")
    public R<ShoppingCart> add(@RequestBody ShoppingCart shoppingCart){
        log.info("购物车数据:{}",shoppingCart);

        //设置用户id,指定当前是哪个用户的购物车数据
        Long currentId = BaseContext.getCurrentId();
        shoppingCart.setUserId(currentId);

        Long dishId = shoppingCart.getDishId();

        LambdaQueryWrapper<ShoppingCart> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(ShoppingCart::getUserId,currentId);

        if(dishId != null){
            //添加到购物车的是菜品
            queryWrapper.eq(ShoppingCart::getDishId,dishId);

        }else{
            //添加到购物车的是套餐
            queryWrapper.eq(ShoppingCart::getSetmealId,shoppingCart.getSetmealId());
        }

        //查询当前菜品或者套餐是否在购物车中
        //SQL:select * from shopping_cart where user_id = ? and dish_id/setmeal_id = ?
        ShoppingCart cartServiceOne = shoppingCartService.getOne(queryWrapper);

        if(cartServiceOne != null){
            //如果已经存在,就在原来数量基础上加一
            Integer number = cartServiceOne.getNumber();
            cartServiceOne.setNumber(number + 1);
            shoppingCartService.updateById(cartServiceOne);
        }else{
            //如果不存在,则添加到购物车,数量默认就是一
            shoppingCart.setNumber(1);
            shoppingCart.setCreateTime(LocalDateTime.now());
            shoppingCartService.save(shoppingCart);
            cartServiceOne = shoppingCart;
        }

        return R.success(cartServiceOne);
    }

3.6购物车---代码开发---查看购物车&清空购物车 97

不同的用户,拥有相互隔离的数据信息,之间不会进行干扰。

本部分实现的效果:在购物车进行查看,添加数量,并且能够看到对应的钱数发生改变。

本部分代码:

    /**
     * 查看购物车
     * @return
     */
    @GetMapping("/list")
    public R<List<ShoppingCart>> list(){
        log.info("查看购物车...");

        LambdaQueryWrapper<ShoppingCart> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(ShoppingCart::getUserId,BaseContext.getCurrentId());//查询条件判断
        queryWrapper.orderByAsc(ShoppingCart::getCreateTime);//根据时间进行降序排序

        List<ShoppingCart> list = shoppingCartService.list(queryWrapper);

        return R.success(list);
    }

本部分实现在购物车部分对菜品的数量进行删减

本部分的代码如下所示:

    /**
     * 清空购物车
     * @return
     */
    @DeleteMapping("/clean")
    public R<String> clean(){
        //SQL: delete from shopping_cart where user_id = ?
        LambdaQueryWrapper<ShoppingCart> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(ShoppingCart::getUserId,BaseContext.getCurrentId());
        
        shoppingCartService.remove(queryWrapper);
        
        return R.success("清空购物车成功");
    }

四、用户下单 98

4.1需求分析 98

实现用户的结账支付。

4.2数据模型 98

 

4.3用户下单---代码开发---梳理交互过程&准备工作 99

移动端处理后的点击“去支付” 功能后发送的请求信息内容。

4.4用户下单---准备工作 99 

4.5用户下单---代码开发 100 

点击“去支付”之后,请求相应的内容如下所示:

如果数据处理过程中,处理的比较复杂,可以在驱动实现的Service服务层进行数据的方法的创建,随后再对应的驱动实现类进行使用。最终在对应的Controller层进行调用使用。

本部分主要

步骤一:实现简单在OrderController层方法搭建submit提交的框架:并调用后面两个步骤的内容的调用。

@Slf4j
@RestController
@RequestMapping("/order")
public class OrderController {

    @Autowired
    private OrderService orderService;


    /**
     * 用户下单
     * @param orders
     * @return
     */
    @PostMapping("/submit")
    public R<String> submit(@RequestBody Orders orders){
        log.info("订单数据:{}",orders);
        orderService.submit(orders);
        return R.success("下单成功");
    }
}

步骤二:

由于操作复杂在service层的接口层进行创建一个新的submit方法

public interface OrderService extends IService<Orders> {

    /**
     * 用户下单
     * @param orders
     */
    public void submit(Orders orders);
}

步骤三:在OrderServiceImpl驱动类实现层进行方法的编写。

4.6用户下单---代码开发(在OrderServiceImpl驱动类实现层实现submit方法实现)101

注本部分是要实现对多个表进行操作处理,需要使用处理多个数据库,即启动AOP事务的运行,所以使用注解@Transactional

实现前面的内容,只有下面的这些内容没有实现

4.7用户下的那---代码开发(向订单表中进行数据的单条插入,订单明细多条数据插入,清空购物车数据)102

1、订单表中数据单条插入

向表中插入数据则需要插入所插入表中的所有数据内容属性,进行编辑设置。

所有在请求中没有反馈的数据,需要在程序代码中进行插入设置。

 代码实现:

@Service
@Slf4j
public class OrderServiceImpl extends ServiceImpl<OrderMapper, Orders> implements OrderService {

    @Autowired
    private ShoppingCartService shoppingCartService;

    @Autowired
    private UserService userService;

    @Autowired
    private AddressBookService addressBookService;

    @Autowired
    private OrderDetailService orderDetailService;

    /**
     * 用户下单
     * @param orders
     */
    @Transactional
    public void submit(Orders orders) {
        //获得当前用户id
        Long userId = BaseContext.getCurrentId();

        //查询当前用户的购物车数据
        LambdaQueryWrapper<ShoppingCart> wrapper = new LambdaQueryWrapper<>();//SQL:查询内容
        wrapper.eq(ShoppingCart::getUserId,userId);
        List<ShoppingCart> shoppingCarts = shoppingCartService.list(wrapper);

        if(shoppingCarts == null || shoppingCarts.size() == 0){//进行异常的报错,当购物车的中没有物品时
            throw new CustomException("购物车为空,不能下单");
        }

        //查询用户数据
        User user = userService.getById(userId);

        //查询地址数据
        Long addressBookId = orders.getAddressBookId();
        AddressBook addressBook = addressBookService.getById(addressBookId);
        if(addressBook == null){
            throw new CustomException("用户地址信息有误,不能下单");
        }

        long orderId = IdWorker.getId();//订单号

        AtomicInteger amount = new AtomicInteger(0); //使用amount来作为购物车计算金额的计算,进行累加的处理
                                                        // 使用原子类型的,保证高并发、多线程时候使用的安全

        //该部分是实现多条数据保存的操作处理,设置为集合List
        List<OrderDetail> orderDetails = shoppingCarts.stream().map((item) -> {
            OrderDetail orderDetail = new OrderDetail();
            orderDetail.setOrderId(orderId);
            orderDetail.setNumber(item.getNumber());
            orderDetail.setDishFlavor(item.getDishFlavor());
            orderDetail.setDishId(item.getDishId());
            orderDetail.setSetmealId(item.getSetmealId());
            orderDetail.setName(item.getName());
            orderDetail.setImage(item.getImage());
            orderDetail.setAmount(item.getAmount());
            amount.addAndGet(item.getAmount().multiply(new BigDecimal(item.getNumber())).intValue());//菜品或者套餐的总金额
            return orderDetail;
        }).collect(Collectors.toList());


        //下面是将Orders类中的属性数据信息进行填充,例如:id、number等等
        orders.setId(orderId);
        orders.setOrderTime(LocalDateTime.now());
        orders.setCheckoutTime(LocalDateTime.now());
        orders.setStatus(2);
        orders.setAmount(new BigDecimal(amount.get()));//总金额
        orders.setUserId(userId);
        orders.setNumber(String.valueOf(orderId));
        orders.setUserName(user.getName());
        orders.setConsignee(addressBook.getConsignee());
        orders.setPhone(addressBook.getPhone());
        orders.setAddress((addressBook.getProvinceName() == null ? "" : addressBook.getProvinceName())
                + (addressBook.getCityName() == null ? "" : addressBook.getCityName())
                + (addressBook.getDistrictName() == null ? "" : addressBook.getDistrictName())
                + (addressBook.getDetail() == null ? "" : addressBook.getDetail()));
        //向订单表插入数据,一条数据
        this.save(orders);

        //向订单明细表插入数据,多条数据
        orderDetailService.saveBatch(orderDetails);

        //清空购物车数据
        shoppingCartService.remove(wrapper);
    }
}

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值