【六】瑞吉外卖开发笔记

  • 个人主页:i笨笨i
  • 版权:本文由【i笨笨i】原创,需要转载联系博主
  • 欢迎关注、点赞、收藏(一键三连)和订阅专栏哦!
  • 一个系列哟
  • 瑞吉外卖开发笔记一
  • 瑞吉外卖开发笔记二
  • 瑞吉外卖开发笔记三
  • 瑞吉外卖开发笔记四
  • 瑞吉外卖开发笔记五

瑞吉外卖开发笔记六

    • 导入用户地址簿相关功能代码
      • 需求分析
      • 数据模型
      • 导入功能代码
      • 功能测试
    • 菜品展示
      • 需求分析
      • 代码开发
      • 功能测试
    • 购物车
      • 需求分析
      • 数据模型
      • 代码开发
        • 交互过程
        • 准备工作
        • 添加购物车
        • 查看购物车
        • 清空购物车
        • 减少菜品
    • 下单
      • 需求分析
      • 数据模型
      • 代码开发
        • 交互过程
        • 准备工作
        • 代码开发
        • 功能测试

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

需求分析

​ 地址簿,指的是移动端消费者用户的地址信息,用户登录成功可以维护自己的地址信息。同一个用户可以有多个地址信息,但是只能有一个默认地址。

【六】瑞吉外卖开发笔记_第1张图片

数据模型

​ 用户的地址信息会存储address_book表,即地址簿表中。具体结构如下:

【六】瑞吉外卖开发笔记_第2张图片

导入功能代码

​ 功能代码清单:

  • AddressBook实体类
  • AddressBookMapper接口类
  • AddressBookService业务层接口
  • AddressBookServiceImpl业务层实现类
  • AddressBookController控制层

1️⃣AddressBook实体类

/**
 * 地址簿
 */
@Data
public class AddressBook implements Serializable {

    private static final long serialVersionUID = 1L;

    private Long id;


    //用户id
    private Long userId;


    //收货人
    private String consignee;


    //手机号
    private String phone;


    //性别 0 女 1 男
    private String sex;


    //省级区划编号
    private String provinceCode;


    //省级名称
    private String provinceName;


    //市级区划编号
    private String cityCode;


    //市级名称
    private String cityName;


    //区级区划编号
    private String districtCode;


    //区级名称
    private String districtName;


    //详细地址
    private String detail;


    //标签
    private String label;

    //是否默认 0 否 1是
    private Integer isDefault;

    //创建时间
    @TableField(fill = FieldFill.INSERT)
    private LocalDateTime createTime;


    //更新时间
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private LocalDateTime updateTime;


    //创建人
    @TableField(fill = FieldFill.INSERT)
    private Long createUser;


    //修改人
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private Long updateUser;


    //是否删除
    private Integer isDeleted;
}

2️⃣AddressBookMapper

@Mapper
public interface AddressBookMapper extends BaseMapper<AddressBook> {
}

3️⃣AddressBookService

public interface AddressBookService extends IService<AddressBook> {
}

4️⃣AddressBookServiceImpl

@Service
public class AddressBookServiceImpl extends ServiceImpl<AddressBookMapper, AddressBook> implements AddressBookService {
}

5️⃣AddressBookController

@Slf4j
@RestController
@RequestMapping("/addressBook")
public class AddressBookController {

    @Autowired
    private AddressBookService addressBookService;

    /**
     * 新增
     *
     * @param addressBook
     * @return
     */
    @PostMapping
    public R<AddressBook> save(@RequestBody AddressBook addressBook) {
        addressBook.setUserId(BaseContext.getCurrentId());
        addressBookService.save(addressBook);
        return R.success(addressBook);
    }

    /**
     * 设置默认地址
     *
     * @param addressBook
     * @return
     */
    @PutMapping("/default")
    public R<AddressBook> setDefault(@RequestBody AddressBook addressBook) {
        LambdaUpdateWrapper<AddressBook> wrapper = new LambdaUpdateWrapper<>();
        wrapper.eq(AddressBook::getUserId, BaseContext.getCurrentId());
        wrapper.set(AddressBook::getIsDefault, 0);
        addressBookService.update(wrapper);

        addressBook.setIsDefault(1);
        addressBookService.updateById(addressBook);
        return R.success(addressBook);
    }

    /**
     * 根据id查询地址
     *
     * @param id
     * @return
     */
    @GetMapping("{/id}")
    public R get(@PathVariable Long id) {
        AddressBook addressBook = addressBookService.getById(id);
        if (addressBook != null) {
            return R.success(addressBook);
        } else {
            return R.error("没有找到该对象");
        }
    }

    /**
     * 查询默认地址
     * @return
     */
    @GetMapping("/default")
    public R<AddressBook> getDefault() {
        LambdaQueryWrapper<AddressBook> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(AddressBook::getUserId, BaseContext.getCurrentId());
        queryWrapper.eq(AddressBook::getIsDefault, 1);

        AddressBook addressBook = addressBookService.getOne(queryWrapper);

        if (null == addressBook){
            return R.error("没有找到该对象");
        }else {
            return R.success(addressBook);
        }
    }

    @GetMapping("/list")
    public R<List<AddressBook>> list(AddressBook addressBook){
        addressBook.setUserId(BaseContext.getCurrentId());
        //条件构造器
        LambdaQueryWrapper<AddressBook> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(null!=addressBook.getUserId(),AddressBook::getUserId,addressBook.getUserId());
        queryWrapper.orderByDesc(AddressBook::getUpdateTime);

        return R.success(addressBookService.list(queryWrapper));
    }
}

功能测试

【六】瑞吉外卖开发笔记_第3张图片

菜品展示

需求分析

​ 用户登录成功后跳转到系统首页,在首页需要根据分类来展示菜品和套餐。如果菜品设置了口味信息需要展示 [选择规格] 按钮,否则显示 [+] 按钮。

【六】瑞吉外卖开发笔记_第4张图片

代码开发

​ 在开发代码之前,需要梳理一下前端页面和服务器的交互过程:

  1. 页面(front/index.html)发送Ajax请求,获取分类数据(菜品分类和套餐分类)
  2. 页面发送Ajax请求,获取第一个分类下的菜品或者套餐

​ 开发菜品展示功能,其实就是在服务端编写代码去处理前端页面发送的这2次请求即可。

1️⃣首页加载完成后,还发送了一次ajax请求用于加载购物车数据,此处可以将这次请求的地址暂时

修改一下,从静态json文件获取数据,等后续开发购物车功能时再修改回来,如下:

//获取购物车内商品的集合
function cartListApi(data) {
    return $axios({
        // 'url': '/shoppingCart/list',
        'url':'/front/cartData.json',
        'method': 'get',
        params:{...data}
    })
}

2️⃣cartData.json

{"code":1,"msg":null,"data":[],"map":{}}

3️⃣改造DishController中的list方法

/**
* 根据条件查询对象的菜品数据
* DishController类中
* @param dish
* @return
*/
@GetMapping("/list")
public R<List<DishDto>> 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);

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

        Long categoryId = item.getCategoryId();

        Category category = categoryService.getById(categoryId);
        if (category != null){
            String categoryName = category.getName();
            dishDto.setCategoryName(categoryName);
        }

        //当前菜品id
        Long dishId = item.getId();
        LambdaQueryWrapper<DishFlavor> lambdaQueryWrapper = new LambdaQueryWrapper<>();
        lambdaQueryWrapper.eq(DishFlavor::getDishId,dishId);
        List<DishFlavor> dishFlavorsList = dishFlavorService.list(lambdaQueryWrapper);
        dishDto.setFlavors(dishFlavorsList);
        return dishDto;
    }).collect(Collectors.toList());

    return R.success(dishDtoList);
}

4️⃣在SetmealController里添加list方法显示套餐信息

/**
* 根据条件查询套餐数据
* SetmealController类中
* @param setmeal
* @return
*/
@GetMapping("/list")
    public R<List<Setmeal>> list(Setmeal setmeal){
    LambdaQueryWrapper<Setmeal> queryWrapper = new LambdaQueryWrapper<>();
    queryWrapper.eq(
            setmeal.getCategoryId()!=null,
            Setmeal::getCategoryId,
            setmeal.getCategoryId());
    queryWrapper.eq(
            setmeal.getStatus()!=null,
            Setmeal::getStatus,
            setmeal.getStatus());
    queryWrapper.orderByDesc(Setmeal::getUpdateTime);

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

    return R.success(list);
}

功能测试

【六】瑞吉外卖开发笔记_第5张图片

购物车

需求分析

​ 移动端用户可以将菜品或者套餐添加到购物车。对于菜品来说,如果设置了口味信息,则需要选择规格后才能加入购物车;对于套餐来说,可以直接点击 [+] 将当前套餐加入购物车。在购物车中可以修改菜品和套餐的数量,也可以清空购物车。

【六】瑞吉外卖开发笔记_第6张图片

数据模型

  • 购物车对应的数据表为shopping_cart表,具体表结构如下:

【六】瑞吉外卖开发笔记_第7张图片

代码开发

交互过程

​ 在开发代码之前,需要梳理一下购物车操作时前端页面和服务端的交互过程:

  1. 点击 [加入购物车] 或者 [+] 按钮,页面发送ajax请求,请求服务端,将菜品或者套餐添加到购物车

  2. 点击购物车图标,页面发送ajax请求,请求服务端查询购物车中的菜品和套餐

  3. 点击清空购物车按钮,页面发送ajax请求,请求服务端来执行清空购物车操作

​ 开发购物车功能,其实就是在服务端编写代码去处理前端页面发送的这3次请求即可。

准备工作

​ 在开发业务功能前,先将需要用到的类和接口基本结构创建:

  • ShoppingCart实体类
  • ShoppingCartMapper接口
  • ShoppingCartService接口
  • ShoppingCartServiceImpl业务层实现
  • ShoppingCartController控制层

1️⃣ShoppingCart实体类

/**
 * 购物车
 */
@Data
public class ShoppingCart implements Serializable {

    private static final long serialVersionUID = 1L;

    private Long id;

    //名称
    private String name;

    //用户id
    private Long userId;

    //菜品id
    private Long dishId;

    //套餐id
    private Long setmealId;

    //口味
    private String dishFlavor;

    //数量
    private Integer number;

    //金额
    private BigDecimal amount;

    //图片
    private String image;

    private LocalDateTime createTime;
}

2️⃣ShoppingCartMapper接口

@Mapper
public interface ShoppingCartMapper extends BaseMapper<ShoppingCart> {
}

3️⃣ShoppingCartService接口

public interface ShoppingCartService extends IService<ShoppingCart> {
}

4️⃣ShoppingCartServiceImpl业务层实现

@Service
public class ShoppingCartServiceImpl extends ServiceImpl<ShoppingCartMapper, ShoppingCart> implements ShoppingCartService {
}

5️⃣ShoppingCartController控制层

@Slf4j
@RestController
@RequestMapping("/shoppingCart")
public class ShoppingCartController {
    @Autowired
    public ShoppingCartService shoppingCartService;
}

添加购物车

    /**
     * 添加购物车
     * ShoppingCartController类中
     * @param shoppingCart
     * @return
     */
@PostMapping("/add")
public R<ShoppingCart> add(@RequestBody ShoppingCart 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());
    }

    //查询当前菜品或者套餐,是否在购物车中
    ShoppingCart cartServiceOne = shoppingCartService.getOne(queryWrapper);

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

    return R.success(cartServiceOne);
}

查看购物车

  • 把前端假数据改回来
function cartListApi(data) {
    return $axios({
        'url': '/shoppingCart/list',
        // 'url':'/front/cartData.json',
        'method': 'get',
        params:{...data}
    })
}
  • 查看购物车
/**
* 查看购物车
* @return
*/
@GetMapping("/list")
public R<List<ShoppingCart>> list(){
    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(){

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

    shoppingCartService.remove(queryWrapper);

    return R.success("清空购物车成功");
}

减少菜品

/**
* 减少菜品
* @param shoppingCart
* @return
*/
@PostMapping("/sub")
public R<ShoppingCart> sub(@RequestBody ShoppingCart shoppingCart){
    Long setmealId = shoppingCart.getSetmealId();
    Long dishId = shoppingCart.getDishId();
    LambdaQueryWrapper<ShoppingCart> queryWrapper = new LambdaQueryWrapper<>();
    queryWrapper.eq(ShoppingCart::getUserId,BaseContext.getCurrentId());

    if (setmealId != null){
        queryWrapper.eq(ShoppingCart::getSetmealId,setmealId);
    }else {
        queryWrapper.eq(ShoppingCart::getDishId,dishId);
    }

    ShoppingCart one = shoppingCartService.getOne(queryWrapper);
    Integer number = one.getNumber();
    if (number == 1){
        shoppingCartService.remove(queryWrapper);
    }else {
        one.setNumber(number-1);
        shoppingCartService.updateById(one);
    }

    return R.success(one);
}

下单

需求分析

​ 移动端用户将菜品或者套餐加入购物车后,可以点击购物车中的 【去结算】 按钮,页面跳转到订单确认页面,点击 【去支付】 按钮则完成下单操作。

【六】瑞吉外卖开发笔记_第8张图片

数据模型

​ 用户下单业务对应的数据表为orders表和order_detail表:

  • orders

【六】瑞吉外卖开发笔记_第9张图片

  • order_detail

【六】瑞吉外卖开发笔记_第10张图片

代码开发

交互过程

​ 在开发代码之前,需要梳理一下用户下单操作时前端页面和服务端的交互过程:

  1. 在购物车中点击 【去结算】 按钮,页面跳转到订单确认页面

  2. 在订单确认页面,发送ajax请求,请求服务端获取当前登录用户的默认地址

  3. 在订单确认页面,发送ajax请求,请求服务端获取当前登录用户的购物车数据

  4. 在订单确认页面点击 【去支付】 按钮,发送ajax请求,请求服务端完成下单操作

​ 开发用户下单功能,其实就是在服务端编写代码去处理前端页面发送的请求即可

准备工作

在开发业务功能前,先将需要用到的类和接口基本结构创建好:

  • 实体类Orders、OrderDetail(直接从课程资料中导入即可)
  • Mapper接口OrderMapper、OrderDetailMapper
  • 业务层接口OrderService、OrderDetailService
  • 业务层实现类OrderServicelmpl、OrderDetailServicelmpl
  • 控制层OrderController、OrderDetailController

1️⃣Orders、OrderDetail实体类

/**
 * 订单
 */
@Data
public class Orders implements Serializable {

    private static final long serialVersionUID = 1L;

    private Long id;

    //订单号
    private String number;

    //订单状态 1待付款,2待派送,3已派送,4已完成,5已取消
    private Integer status;


    //下单用户id
    private Long userId;

    //地址id
    private Long addressBookId;


    //下单时间
    private LocalDateTime orderTime;


    //结账时间
    private LocalDateTime checkoutTime;


    //支付方式 1微信,2支付宝
    private Integer payMethod;


    //实收金额
    private BigDecimal amount;

    //备注
    private String remark;

    //用户名
    private String userName;

    //手机号
    private String phone;

    //地址
    private String address;

    //收货人
    private String consignee;
}
/**
 * 订单明细
 */
@Data
public class OrderDetail implements Serializable {

    private static final long serialVersionUID = 1L;

    private Long id;

    //名称
    private String name;

    //订单id
    private Long orderId;


    //菜品id
    private Long dishId;


    //套餐id
    private Long setmealId;


    //口味
    private String dishFlavor;


    //数量
    private Integer number;

    //金额
    private BigDecimal amount;

    //图片
    private String image;
}

2️⃣OrderMapper、OrderDetailMapper接口

@Mapper
public interface OrderMapper extends BaseMapper<Orders> {
}
@Mapper
public interface OrderDetailMapper extends BaseMapper<OrderDetail> {
}

3️⃣OrderService、OrderDetailService业务层接口

public interface OrderService extends IService<Orders> {
}
public interface OrderDetailService extends IService<OrderDetail> {
}

4️⃣OrderServicelmpl、OrderDetailServicelmpl业务层是实现类

@Service
public class OrderServiceImpl extends ServiceImpl<OrderMapper, Orders> implements OrderService {
}
@Service
public class OrderDetailServiceImpl extends ServiceImpl<OrderDetailMapper, OrderDetail> implements OrderDetailService {
}

5️⃣OrderController、OrderDetailController控制层

/**
 * 订单
 */
@Slf4j
@RestController
@RequestMapping("/order")
public class OrderController {
    @Autowired
    private OrderService orderService;
}
@Slf4j
@RestController
@RequestMapping("/orderDetail")
public class OrderDetailController {
    @Autowired
    private OrderDetailService orderDetailService;
}

代码开发

1️⃣在OrderService添加submit方法用于用户下单

  • OrderService业务接口
/**
* 用户下单
* @param orders
*/
void submit(Orders orders);
  • 实现类
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.stream.Collectors;

@Service
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
     */
    @Override
    @Transactional
    public void submit(Orders orders) {
        //获取当前用户id
        Long userId = BaseContext.getCurrentId();

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

        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);

        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());

        //向订单表插入数据(1条),
        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.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(queryWrapper);
    }
}

2️⃣在OrderController的submit方法处理post请求实现上面的方法

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

功能测试

  • 下单界面

【六】瑞吉外卖开发笔记_第11张图片

  • 下单成功界面

【六】瑞吉外卖开发笔记_第12张图片

你可能感兴趣的:(瑞吉外卖项目,java,servlet,mybatis,spring,maven)