【苍穹外卖】Day04 套餐管理

新增套餐

页面以及业务规则

业务规则:

  • 套餐名称唯一
  • 套餐必须属于某个分类
  • 套餐必须包含菜品
  • 名称、分类、价格、图片为必填项
  • 添加菜品窗口需要根据分类类型来展示菜品
  • 新增的套餐默认为停售状态

接口设计(共涉及到4个接口):

  • 根据类型查询分类(已完成)
  • 根据分类id查询菜品
  • 图片上传(已完成)
  • 新增套餐

代码实现

首先,与添加菜品一样的逻辑,前端需要先向后端发送查询分类的请求,会根据类型返回,类型一共只有菜品分类和套餐分类俩种类型。这里后端会根据类型查询这个类别下有哪些类别,并返回给前端。用到的代码与菜品管理章节中是一致的,详细代码可以看分类管理章节,代码在那个章节实现。

其次,点击添加菜品的时候,需要根据分类查询当前分类下面有哪些菜品

复制代码
/**
 * 根据分类id查询菜品
 * @param categoryId
 * @return
 */
@GetMapping("/list")
@ApiOperation("根据分类id查询菜品")
public Result<List<Dish>> list(Long categoryId){
    List<Dish> list = dishService.list(categoryId);
    return Result.success(list);
}

//===========================================================

/**
 * 根据分类id查询菜品
 * @param categoryId
 * @return
 */
public List<Dish> list(Long categoryId) {
    Dish dish = Dish.builder()
            .categoryId(categoryId)
            .status(StatusConstant.ENABLE)
            .build();
    return dishMapper.list(dish);
}

//===========================================================
<select id="list" resultType="Dish" parameterType="Dish">
    select * from dish
    <where>
        <if test="name != null">
            and name like concat('%',#{name},'%')
        </if>
        <if test="categoryId != null">
            and category_id = #{categoryId}
        </if>
        <if test="status != null">
            and status = #{status}
        </if>
    </where>
    order by create_time desc
</select>

用户点击不同的菜品分类,查询相应的菜品,并用一个List集合包含返回给前端,这样前端在左侧类别中点击不同类别会显示不同的菜品。如下图:

第三步:用户需要给当前套餐上传图片,这里用到的代码逻辑请回顾菜品管理。

最后一步,用户将上述信息都填写完毕后,点击保存并继续添加按钮,会触发套餐的增加。前端发送请求,后端进行处理。因为套餐会包含菜品,所以操作的时候会操作多个数据库表单,所以我们要加上业务注解@Transactional。

复制代码
/**
 * 新增套餐
 * @param setmealDTO
 * @return
 */
@PostMapping
@ApiOperation("新增套餐")
@CacheEvict(cacheNames = "setmealCache", key = "#setmealDTO.categoryId")
public Result save(@RequestBody SetmealDTO setmealDTO) {
    setmealService.saveWithDish(setmealDTO);
    return Result.success();
}

//======================================================

/**
 * 新增套餐,同时需要保存套餐和菜品的关联关系
 *
 * @param setmealDTO
 */
@Transactional
public void saveWithDish(SetmealDTO setmealDTO) {
    Setmeal setmeal = new Setmeal();
    BeanUtils.copyProperties(setmealDTO, setmeal);

    //向套餐表插入数据
    setmealMapper.insert(setmeal);

    //获取生成的套餐id
    Long setmealId = setmeal.getId();

    List<SetmealDish> setmealDishes = setmealDTO.getSetmealDishes();
    setmealDishes.forEach(setmealDish -> {
        setmealDish.setSetmealId(setmealId);
    });

    //保存套餐和菜品的关联关系
    setmealDishMapper.insertBatch(setmealDishes);
}

套餐分页查询

业务规则:

  • 根据页码进行分页展示
  • 每页展示10条数据
  • 可以根据需要,按照套餐名称、分类、售卖状态进行查询

代码实现

分页查询,我们依旧使用的是pagehelper来实现,在SQL内部进行额外条件的查询。

复制代码
/**
 * 分页查询
 * @param setmealPageQueryDTO
 * @return
 */
@GetMapping("/page")
@ApiOperation("分页查询")
public Result<PageResult> page(SetmealPageQueryDTO setmealPageQueryDTO) {
PageResult pageResult = setmealService.pageQuery(setmealPageQueryDTO);
return Result.success(pageResult);
}

//=======================================================

/**
 * 分页查询
 *
 * @param setmealPageQueryDTO
 * @return
 */
public PageResult pageQuery(SetmealPageQueryDTO setmealPageQueryDTO) {
    int pageNum = setmealPageQueryDTO.getPage();
    int pageSize = setmealPageQueryDTO.getPageSize();

    PageHelper.startPage(pageNum, pageSize);
    Page<SetmealVO> page = setmealMapper.pageQuery(setmealPageQueryDTO);
    return new PageResult(page.getTotal(), page.getResult());
}

//========================================================

<select id="pageQuery" resultType="com.sky.vo.SetmealVO">
    select
    s.*,c.name categoryName
    from
    setmeal s
    left join
    category c
    on
    s.category_id = c.id
    <where>
        <if test="name != null">
            and s.name like concat('%',#{name},'%')
        </if>
        <if test="status != null">
            and s.status = #{status}
        </if>
        <if test="categoryId != null">
            and s.category_id = #{categoryId}
        </if>
    </where>
    order by s.create_time desc
</select>

删除套餐

类似于菜品管理,允许前端传入多个id进行批量删除,这里再进行赘述。

复制代码
/**
 * 批量删除套餐
 * @param ids
 * @return
 */
@DeleteMapping
@ApiOperation("批量删除套餐")
@CacheEvict(cacheNames = "setmealCache", allEntries = true)
public Result delete(@RequestParam List<Long> ids){
    setmealService.deleteBatch(ids);
    return Result.success();
}

修改套餐

页面以及业务规则

我们需要完成俩个接口:根据id查询套餐信息,修改套餐信息

代码实现

首先是查询套餐信息,前端传递当前套餐id,后端根据id查询套餐,并且还需要到套餐与菜品关联表,查询关联菜品信息,最后将俩个结果组合返回即可。

复制代码
/**
 * 根据id查询套餐和套餐菜品关系
 *
 * @param id
 * @return
 */
public SetmealVO getByIdWithDish(Long id) {
    Setmeal setmeal = setmealMapper.getById(id);
    List<SetmealDish> setmealDishes = setmealDishMapper.getBySetmealId(id);

    SetmealVO setmealVO = new SetmealVO();
    BeanUtils.copyProperties(setmeal, setmealVO);
    setmealVO.setSetmealDishes(setmealDishes);

    return setmealVO;
}

随着信息填写完毕,点击修改后,再触发修改操作。类似的操作在菜品管理也有。

复制代码
/**
 * 修改套餐
 *
 * @param setmealDTO
 */
@Transactional
public void update(SetmealDTO setmealDTO) {
    Setmeal setmeal = new Setmeal();
    BeanUtils.copyProperties(setmealDTO, setmeal);

    //1、修改套餐表,执行update
    setmealMapper.update(setmeal);

    //套餐id
    Long setmealId = setmealDTO.getId();

    //2、删除套餐和菜品的关联关系,操作setmeal_dish表,执行delete
    setmealDishMapper.deleteBySetmealId(setmealId);

    List<SetmealDish> setmealDishes = setmealDTO.getSetmealDishes();
    setmealDishes.forEach(setmealDish -> {
        setmealDish.setSetmealId(setmealId);
    });
    //3、重新插入套餐和菜品的关联关系,操作setmeal_dish表,执行insert
    setmealDishMapper.insertBatch(setmealDishes);
}

起售停售套餐

类似于菜品管理。不过起售套餐需要多加一层判断,判断套餐内是否有停售菜品,有停售菜品提示"套餐内包含未启售菜品,无法启售"。

复制代码
/**
 * 套餐起售、停售
 * @param status
 * @param id
 */
public void startOrStop(Integer status, Long id) {
    //起售套餐时,判断套餐内是否有停售菜品,有停售菜品提示"套餐内包含未启售菜品,无法启售"
    if(status == StatusConstant.ENABLE){
        //select a.* from dish a left join setmeal_dish b on a.id = b.dish_id where b.setmeal_id = ?
        List<Dish> dishList = dishMapper.getBySetmealId(id);
        if(dishList != null && dishList.size() > 0){
            dishList.forEach(dish -> {
                if(StatusConstant.DISABLE == dish.getStatus()){
                    throw new SetmealEnableFailedException(MessageConstant.SETMEAL_ENABLE_FAILED);
                }
            });
        }
    }

    Setmeal setmeal = Setmeal.builder()
            .id(id)
            .status(status)
            .build();
    setmealMapper.update(setmeal);
}
相关推荐
前端不太难17 小时前
HarmonyOS PC 应用的维护成本,从哪来?
状态模式·harmonyos
CryptoRzz1 天前
如何高效接入日本股市实时数据?StockTV API 对接实战指南
java·python·kafka·区块链·状态模式·百度小程序
全栈技术负责人1 天前
前端团队 AI Core Workflow:从心法到落地
前端·人工智能·状态模式
前端不太难2 天前
HarmonyOS PC 多窗口,到底在解决什么
华为·状态模式·harmonyos
a程序小傲3 天前
京东Java面试被问:基于Gossip协议的最终一致性实现和收敛时间
java·开发语言·前端·数据库·python·面试·状态模式
橙序员小站3 天前
走向全栈:前后端状态认知差异与设计边界的深度探讨
状态模式
Beginner x_u3 天前
前端八股文 Vue下
前端·vue.js·状态模式
apolloyhl4 天前
State 状态模式
状态模式
冬奇Lab4 天前
【Kotlin系列14】编译器插件与注解处理器开发:在编译期操控Kotlin
android·开发语言·kotlin·状态模式