6.套餐管理业务开发

套餐管理业务开发

1、准备工作

准备需要用到的类和接口

  1. 实体类SetmealDish
  2. DTO SetmealDto
  3. Mapper接口SetmealDishMapper
  4. 业务层接口SetmealDishService
  5. 业务层实现类SetmealDishServiceImpl
  6. 控制层SetmealController

2、新增套餐

执行流程:

  1. 页面(backend/page/comboladd.html)发送ajax请求,请求服务端获取套餐分类数据并展示到下拉框中
  2. 页面发送ajax请求,请求服务端获取菜品分类数据并展示到添加菜品窗口中
  3. 页面发送ajax请求,请求服务端,根据菜品分类查询对应的菜品数据并展示到添加菜品窗口中
  4. 页面发送请求进行图片上传,请求服务端将图片保存到服务器
  5. 页面发送请求进行图片下载,将上传的图片进行回显
  6. 点击保存按钮,发送ajax请求,将套餐相关数据以json形式提交到服务端

2.1、查询菜品分类列表

DishController创建list方法

/**
 * 根据条件查询对应的菜品数据
 * @param dish
 * @return
 */
@GetMapping("/list")
public R<List<Dish>> list(Dish dish) {

    // 狗仔查询条件
    LambdaQueryWrapper<Dish> queryWrapper = new LambdaQueryWrapper<>();
    queryWrapper.eq(dish.getCategoryId() != null, Dish::getCategoryId, dish.getCategoryId());
    // 查询状态为1的菜品
    queryWrapper.eq(Dish::getStatus, 1);

    // 添加排序条件
    queryWrapper.orderByAsc(Dish::getSort).orderByDesc(Dish::getUpdateTime);

    List<Dish> list = dishService.list(queryWrapper);

    return R.success(list);
}

2.2、阿里云OSS图片回显(可选)

修改backend/page/combo/add.html页面以下内容,改成自己的阿里云地址加上图片名称

image-20220619153241714

2.3、保存新增信息

2.3.1、SetmealController
package com.itheima.reggie.controller;

import com.itheima.reggie.common.R;
import com.itheima.reggie.dto.SetmealDto;
import com.itheima.reggie.entity.SetmealDish;
import com.itheima.reggie.service.SetmealDishService;
import com.itheima.reggie.service.SetmealService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

/**
 * @Author: xjhqre
 * @DateTime: 2022/6/19 15:36
 */
@RequestMapping("/setmeal")
@RestController
@Slf4j
public class SetmealController {

    @Autowired
    SetmealService setmealService;

    @Autowired
    SetmealDishService setmealDishService;

    /**
     * 新增套餐
     * @param setmealDto
     * @return
     */
    @PostMapping
    public R<String> save(@RequestBody SetmealDto setmealDto) {
        log.info("新增套餐信息:{}", setmealDto);
        setmealService.saveWithDish(setmealDto);
        return R.success("添加套餐成功");
    }
}
2.3.2、SetmealService
// 新增套餐,同时需要保存套餐和菜品的关联关系
void saveWithDish(SetmealDto setmealDto);
2.3.3、SetmealServiceImpl
/**
 * 新增套餐,同时需要保存套餐和菜品的关联关系
 * @param setmealDto
 */
@Override
@Transactional
public void saveWithDish(SetmealDto setmealDto) {
    // 保存套餐的基本信息,操作setmeal,执行insert操作
    this.save(setmealDto);

    List<SetmealDish> setmealDishes = setmealDto.getSetmealDishes();
    setmealDishes = setmealDishes.stream().map(item -> {
        item.setSetmealId(setmealDto.getId());
        return item;
    }).collect(Collectors.toList());

    // 保存套餐和菜品的关联信息,操作setmeal_dish执行insert操作
    setmealDishService.saveBatch(setmealDishes);
}

2.4、测试新增套餐

3、套餐信息分页查询

执行流程:

  1. 页面(backend/page/combo/list.html)发送ajax请求,将分页查询参数(page、pageSize.name)提交到服务端,获取分页数据
  2. 页面发送请求,请求服务端进行图片下载,用于页面图片展示

3.1、方法一:老师的方法

SetmealController创建page方法

/**
 * 套餐信息分页查询
 * @param page
 * @param pageSize
 * @param name
 * @return
 */
@GetMapping("/page")
public R<Page> page(int page, int pageSize, String name) {
    // 分页构造器对象
    Page<Setmeal> pageInfo = new Page<>(page, pageSize);
    Page<SetmealDto> dtoPage = new Page<>();

    LambdaQueryWrapper<Setmeal> queryWrapper = new LambdaQueryWrapper<>();
    // 添加查询条件,根据name进行like模糊查询
    queryWrapper.like(name != null, Setmeal::getName, name);
    // 添加排序条件,根据更新时间降序排序
    queryWrapper.orderByDesc(Setmeal::getUpdateTime);

    setmealService.page(pageInfo, queryWrapper);

    // 对象拷贝
    BeanUtils.copyProperties(pageInfo, dtoPage, "records");
    List<Setmeal> records = pageInfo.getRecords();

    List<SetmealDto> list = records.stream().map(item -> {
        SetmealDto setmealDto = new SetmealDto();
        // 对象拷贝
        BeanUtils.copyProperties(item, setmealDto);
        // 分类id
        Long categoryId = item.getCategoryId();
        // 根据分类id查询分类对象
        Category category = categoryService.getById(categoryId);
        if (category != null) {
            // 分类名称
            String categoryName = category.getName();
            setmealDto.setCategoryName(categoryName);
        }
        return setmealDto;
    }).collect(Collectors.toList());

    dtoPage.setRecords(list);
    return R.success(pageInfo);
}

3.2、方法二:联表查询方法

3.2.1、SetmealController
/**
 * 套餐信息分页查询
 * @param page
 * @param pageSize
 * @param name
 * @return
 */
@GetMapping("/page")
public R<Page<SetmealDto>> page(int page, int pageSize, String name) {
    // 分页构造器
    Page<SetmealDto> pageInfo = new Page<>(page, pageSize);

    setmealService.querySetmealPageWithCategoryName(pageInfo, name);

    return R.success(pageInfo);
}
3.2.2、SetmealService
// 查询带分类名的套餐分页信息
void querySetmealPageWithCategoryName(Page<SetmealDto> pageInfo, String name);
3.2.3、SetmealServiceImpl
/**
 * 查询带分类名的套餐分页信息
 * @param pageInfo
 * @param name
 */
@Override
public void querySetmealPageWithCategoryName(Page<SetmealDto> pageInfo, String name) {
    List<SetmealDto> records = setmealMapper.querySetmealPageWithCategoryName(pageInfo, name == null ? "" : name);
    pageInfo.setRecords(records);
}
3.2.4、SetmealMapper
@Select("SELECT a.*, b.`name` AS category_name FROM setmeal a JOIN category b ON a.`category_id` = b.`id` WHERE a.`name` LIKE CONCAT('%',#{name},'%') AND a.`is_deleted` = 0 ORDER BY sort")
List<SetmealDto> querySetmealPageWithCategoryName(Page<SetmealDto> pageInfo, String name);

3.3、套餐管理页面阿里云OSS图片显示

修改backend/page/combo/list.html文件以下两处地方

image-20220619171621422

image-20220619171653708

3.4、测试分页查询

image-20220619173331821

4、删除套餐信息

  1. 前端发送删除请求,参数形式为ids=id1,id2,...
  2. 后端接收请求,根据id先删除表setmeal_dish表的内容,在删除setmeal表的内容

删除方法为逻辑删除

4.1 修改Setmeal和SetmealDish实体类

修改两个实体类的isDeleted属性,加上@TableLogic(value = "0", delval = "1")注解。用于逻辑删除。标注了此注解后,调用mybatis-plus框架里给的删除方法,会转变成对is_deleted字段的更新方法

//是否删除
@TableLogic(value = "0", delval = "1")
private Integer isDeleted;

4.2、SetmealController

创建delete方法

/**
 * 删除套餐
 * @param ids
 * @return
 */
@DeleteMapping
public R<String> delete(@RequestParam Long[] ids) {
    log.info("要删除套餐的ids:{}", Arrays.toString(ids));
    
    setmealService.removeSetmealWithDish(ids);
    
    return R.success("套餐数据删除成功");
}

4.3、SetmealService

// 删除套餐,同时需要删除套餐和菜品的关联数据
void removeSetmealWithDish(Long[] ids);

4.4、SetmealServiceImpl

/**
 * 删除套餐,同时需要删除套餐和菜品的关联数据
 * @param ids
 */
@Override
@Transactional
public void removeSetmealWithDish(Long[] ids) {
    // 查询套餐状态,确定是否可用删除
    LambdaQueryWrapper<Setmeal> queryWrapper = new LambdaQueryWrapper<>();
    queryWrapper.in(Setmeal::getId, ids);
    queryWrapper.eq(Setmeal::getStatus, 1);
    int count = this.count(queryWrapper);
    if (count > 0) {
        // 当前套餐正在出售,不能删除
        throw new CustomException("套餐正在售卖中,不能删除");
    }

    // 如果可用删除,先删除表setmeal_dish中关联的菜品信息
    // delete from setmeal_dish where setmeal_id in ()
    LambdaQueryWrapper<SetmealDish> setmealDishLambdaQueryWrapper = new LambdaQueryWrapper<>();
    setmealDishLambdaQueryWrapper.in(SetmealDish::getSetmealId, ids);
    setmealDishService.remove(setmealDishLambdaQueryWrapper);

    // 然后删除套餐信息
    this.removeByIds(Arrays.asList(ids));
}

4.5、测试删除套餐

5、修改套餐信息(可选)

5.1、套餐数据回显

点击修改后,前端页面会发送一个查询请求,带有id参数

image-20220619195308895

我们需要返回一个SetmealDto类型的对象

5.1.1、SetmealController
/**
 * 根据id查询套餐信息和对应的菜品信息
 * @param id
 * @return
 */
@GetMapping("/{id}")
public R<SetmealDto> get(@PathVariable Long id) {
    SetmealDto setmealDto = setmealService.getByIdWithDish(id);
    return R.success(setmealDto);
}
5.1.2、SetmealService
// 根据id查询套餐信息和对应的菜品信息
SetmealDto getByIdWithDish(Long id);
5.1.3、SetmealServiceImpl
/**
 * 根据id查询套餐信息和对应的菜品信息
 * @param id
 * @return
 */
@Override
@Transactional
public SetmealDto getByIdWithDish(Long id) {
    // 查询套餐基本信息
    Setmeal setmeal = this.getById(id);

    SetmealDto setmealDto = new SetmealDto();
    BeanUtils.copyProperties(setmeal, setmealDto);

    // 查询当前套餐对应的菜品信息
    // select * from setmeal_dish where setmeal_id = id;
    LambdaQueryWrapper<SetmealDish> queryWrapper = new LambdaQueryWrapper<>();
    queryWrapper.eq(SetmealDish::getSetmealId, id);
    List<SetmealDish> setmealDishes = setmealDishService.list(queryWrapper);

    setmealDto.setSetmealDishes(setmealDishes);

    return setmealDto;
}
5.1.4、阿里云OSS图片回显(可选)

修改backend/page/combo/add.html文件以下内容

image-20220619200351111

5.1.5、测试数据回显

5.2、保存修改信息

先删除setmeal对应的所有dish,在进行添加

5.2.1、SetmealController
/**
 * 修改套餐
 * @param setmealDto
 * @return
 */
@PutMapping
public R<String> update(@RequestBody SetmealDto setmealDto) {
    log.info("保存修改套餐信息:{}", setmealDto);
    setmealService.updateSetmealWithDish(setmealDto);
    return R.success("修改套餐成功");
}
5.2.2、SetmealService
// 更新套餐信息
void updateSetmealWithDish(SetmealDto setmealDto);
5.2.3、SetmealServiceImpl
/**
 * 更新套餐信息
 * @param setmealDto
 */
@Override
public void updateSetmealWithDish(SetmealDto setmealDto) {
    // 更新setmeal表基本信息
    this.updateById(setmealDto);

    // 删除原来对应的菜品信息
    LambdaQueryWrapper<SetmealDish> queryWrapper = new LambdaQueryWrapper<>();
    queryWrapper.eq(SetmealDish::getSetmealId, setmealDto.getId());
    setmealDishService.remove(queryWrapper);

    // 添加新的菜品信息
    List<SetmealDish> setmealDishes = setmealDto.getSetmealDishes();
    setmealDishes = setmealDishes.stream().peek(item -> item.setSetmealId(setmealDto.getId())).collect(Collectors.toList());
    setmealDishService.saveBatch(setmealDishes);
}
5.2.4、测试修改套餐

6、批量起售/停售套餐(可选)

前端页面请求:

image-20220619215306929

6.1、SetmealController

/**
 * 批量起售/停售套餐
 * @param status
 * @param ids
 * @return
 */
@PostMapping("/status/{status}")
public R<String> status(@PathVariable("status") int status, @RequestParam Long[] ids) {
    log.info("批量起售停售:{}", status, Arrays.toString(ids));
    setmealService.updateStatusByIds(status, ids);
    return R.success("批量起售停售成功");
}

6.2、SetmealService

// 批量起售停售套餐
void updateStatusByIds(int status, Long[] ids);

6.3、SetmealServiceImpl

不知道为什么mybat-plus没有提供批量修改的方法。

这里我为了方便就直接在循环里更新了。

还有一种方法是在mapper文件里进行批量更新。

不知道哪种方法更优雅一点。如果有大佬知道希望能在评论区解答一下。

/**
 * 批量起售停售套餐
 * @param status
 * @param ids
 */
@Override
public void updateStatusByIds(int status, Long[] ids) {
    // 查询出所有对应id的setmeal
    List<Setmeal> setmeals = setmealMapper.selectBatchIds(Arrays.asList(ids));

    // 循环遍历更新
    setmeals.forEach(item -> {
        item.setStatus(status);
        setmealMapper.updateById(item);
    });
}

6.4、测试批量起售停售

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值