太原网站建设与维护,秦皇岛建设局,动漫设计专升本考什么,伍佰亿网站怎样查询历史订单——需求分析与设计
产品原型 业务规则 分页查询历史订单 可以根据订单状态查询 展示订单数据时#xff0c;需要展示的数据包括#xff1a;下单时间、订单状态、订单金额、订单明细#xff08;商品名称、图片#xff09;
接口设计 查询历史订单——代码开…查询历史订单——需求分析与设计
产品原型 业务规则 分页查询历史订单 可以根据订单状态查询 展示订单数据时需要展示的数据包括下单时间、订单状态、订单金额、订单明细商品名称、图片
接口设计 查询历史订单——代码开发
Controller层中 /*** 历史订单查询** param page* param pageSize* param status 订单状态 1待付款 2待接单 3已接单 4派送中 5已完成 6已取消* return*/GetMapping(/historyOrders)ApiOperation(历史订单查询)public ResultPageResult page(int page, int pageSize, Integer status) {PageResult pageResult orderService.pageQuery4User(page, pageSize, status);return Result.success(pageResult);}
Service层中
/*** 用户端订单分页查询** param pageNum* param pageSize* param status* return*/public PageResult pageQuery4User(int pageNum, int pageSize, Integer status) {// 设置分页PageHelper.startPage(pageNum, pageSize);OrdersPageQueryDTO ordersPageQueryDTO new OrdersPageQueryDTO();ordersPageQueryDTO.setUserId(BaseContext.getCurrentId());ordersPageQueryDTO.setStatus(status);// 分页条件查询PageOrders page orderMapper.pageQuery(ordersPageQueryDTO);ListOrderVO list new ArrayList();// 查询出订单明细并封装入OrderVO进行响应if (page ! null page.getTotal() 0) {for (Orders orders : page) {Long orderId orders.getId();// 订单id// 查询订单明细ListOrderDetail orderDetails orderDetailMapper.getByOrderId(orderId);OrderVO orderVO new OrderVO();BeanUtils.copyProperties(orders, orderVO);orderVO.setOrderDetailList(orderDetails);list.add(orderVO);}}return new PageResult(page.getTotal(), list);}
Mapper层中
OrderMapper /*** 分页条件查询并按下单时间排序* param ordersPageQueryDTO*/PageOrders pageQuery(OrdersPageQueryDTO ordersPageQueryDTO);
对应的映射文件 select idpageQuery resultTypeOrdersselect * from orderswhereif testnumber ! null and number!and number like concat(%,#{number},%)/ifif testphone ! null and phone!and phone like concat(%,#{phone},%)/ifif testuserId ! nulland user_id #{userId}/ifif teststatus ! nulland status #{status}/ifif testbeginTime ! nulland order_time gt; #{beginTime}/ifif testendTime ! nulland order_time lt; #{endTime}/if/whereorder by order_time desc/select OrderDetailMapper /*** 根据订单id查询订单明细* param orderId* return*/Select(select * from order_detail where order_id #{orderId})ListOrderDetail getByOrderId(Long orderId);
查询历史订单——功能测试
测试无误 查询订单详情——需求分析与设计
产品原型 接口设计 查询订单详情——代码开发
Controller层中 /*** 查询订单详情** param id* return*/GetMapping(/orderDetail/{id})ApiOperation(查询订单详情)public ResultOrderVO details(PathVariable(id) Long id) {OrderVO orderVO orderService.details(id);return Result.success(orderVO);}
Service层中 /*** 查询订单详情** param id* return*/public OrderVO details(Long id) {// 根据id查询订单Orders orders orderMapper.getById(id);// 查询该订单对应的菜品/套餐明细ListOrderDetail orderDetailList orderDetailMapper.getByOrderId(orders.getId());// 将该订单及其详情封装到OrderVO并返回OrderVO orderVO new OrderVO();BeanUtils.copyProperties(orders, orderVO);orderVO.setOrderDetailList(orderDetailList);return orderVO;}
Mapper层中 /*** 根据id查询订单* param id*/Select(select * from orders where id#{id})Orders getById(Long id); 查询订单详情——功能测试 取消订单——需求分析与设计
产品原型 业务规则 待支付和待接单状态下用户可直接取消订单 商家已接单状态下用户取消订单需电话沟通商家 派送中状态下用户取消订单需电话沟通商家 如果在待接单状态下取消订单需要给用户退款 取消订单后需要将订单状态修改为“已取消”
接口设计 取消订单——代码开发
Controller层中 /*** 用户取消订单** return*/PutMapping(/cancel/{id})ApiOperation(取消订单)public Result cancel(PathVariable(id) Long id) throws Exception {orderService.userCancelById(id);return Result.success();}
Service层中 /*** 用户取消订单** param id*/public void userCancelById(Long id) throws Exception {// 根据id查询订单Orders ordersDB orderMapper.getById(id);// 校验订单是否存在if (ordersDB null) {throw new OrderBusinessException(MessageConstant.ORDER_NOT_FOUND);}//订单状态 1待付款 2待接单 3已接单 4派送中 5已完成 6已取消if (ordersDB.getStatus() 2) {throw new OrderBusinessException(MessageConstant.ORDER_STATUS_ERROR);}Orders orders new Orders();orders.setId(ordersDB.getId());// 订单处于待接单状态下取消需要进行退款if (ordersDB.getStatus().equals(Orders.TO_BE_CONFIRMED)) {//调用微信支付退款接口weChatPayUtil.refund(ordersDB.getNumber(), //商户订单号ordersDB.getNumber(), //商户退款单号new BigDecimal(0.01),//退款金额单位 元new BigDecimal(0.01));//原订单金额//支付状态修改为 退款orders.setPayStatus(Orders.REFUND);}// 更新订单状态、取消原因、取消时间orders.setStatus(Orders.CANCELLED);orders.setCancelReason(用户取消);orders.setCancelTime(LocalDateTime.now());orderMapper.update(orders);}
Mapper层中
没有新东西。 取消订单——功能测试
因没有搞好支付那块没法在已付款待加单下测试。 再来一单——需求分析与设计
产品原型 接口设计 业务规则 再来一单就是将原订单中的商品重新加入到购物车中 再来一单——代码开发
Controller层中 /*** 再来一单** param id* return*/PostMapping(/repetition/{id})ApiOperation(再来一单)public Result repetition(PathVariable Long id) {orderService.repetition(id);return Result.success();}
Service层中 /*** 再来一单** param id*/public void repetition(Long id) {// 查询当前用户idLong userId BaseContext.getCurrentId();// 根据订单id查询当前订单详情ListOrderDetail orderDetailList orderDetailMapper.getByOrderId(id);// 将订单详情对象转换为购物车对象ListShoppingCart shoppingCartList orderDetailList.stream().map(x - {ShoppingCart shoppingCart new ShoppingCart();// 将原订单详情里面的菜品信息重新复制到购物车对象中BeanUtils.copyProperties(x, shoppingCart, id);shoppingCart.setUserId(userId);shoppingCart.setCreateTime(LocalDateTime.now());return shoppingCart;}).collect(Collectors.toList());// 将购物车对象批量添加到数据库shoppingCartMapper.insertBatch(shoppingCartList);}
Mapper层中
ShoppingCartMapper /*** 批量插入购物车数据** param shoppingCartList*/void insertBatch(ListShoppingCart shoppingCartList);
对应的映射文件中
insert idinsertBatch parameterTypelistinsert into shopping_cart(name, image, user_id, dish_id, setmeal_id, dish_flavor, number, amount, create_time)valuesforeach collectionshoppingCartList itemsc separator,(#{sc.name},#{sc.image},#{sc.userId},#{sc.dishId},#{sc.setmealId},#{sc.dishFlavor},#{sc.number},#{sc.amount},#{sc.createTime})/foreach
/insert 再来一单——功能测试
测试无误
接下来都是管理端的代码
接下来都是管理端的代码
接下来都是管理端的代码
接下来都是管理端的代码
订单搜索——需求分析与设计
产品原型 业务规则
- 输入订单号/手机号进行搜索支持模糊搜索 - 根据订单状态进行筛选 - 下单时间进行时间筛选 - 搜索内容为空提示未找到相关订单 - 搜索结果页展示包含搜索关键词的内容 - 分页展示搜索到的订单数据
接口设计 订单搜索——代码开发
Controller层中
/*** 订单管理*/
RestController(adminOrderController)
RequestMapping(/admin/order)
Slf4j
Api(tags 订单管理接口)
public class OrderController {Autowiredprivate OrderService orderService;/*** 订单搜索** param ordersPageQueryDTO* return*/GetMapping(/conditionSearch)ApiOperation(订单搜索)public ResultPageResult conditionSearch(OrdersPageQueryDTO ordersPageQueryDTO) {PageResult pageResult orderService.conditionSearch(ordersPageQueryDTO);return Result.success(pageResult);}
}
Service层中 /*** 订单搜索** param ordersPageQueryDTO* return*/public PageResult conditionSearch(OrdersPageQueryDTO ordersPageQueryDTO) {PageHelper.startPage(ordersPageQueryDTO.getPage(), ordersPageQueryDTO.getPageSize());PageOrders page orderMapper.pageQuery(ordersPageQueryDTO);// 部分订单状态需要额外返回订单菜品信息将Orders转化为OrderVOListOrderVO orderVOList getOrderVOList(page);return new PageResult(page.getTotal(), orderVOList);}private ListOrderVO getOrderVOList(PageOrders page) {// 需要返回订单菜品信息自定义OrderVO响应结果ListOrderVO orderVOList new ArrayList();ListOrders ordersList page.getResult();if (!CollectionUtils.isEmpty(ordersList)) {for (Orders orders : ordersList) {// 将共同字段复制到OrderVOOrderVO orderVO new OrderVO();BeanUtils.copyProperties(orders, orderVO);String orderDishes getOrderDishesStr(orders);// 将订单菜品信息封装到orderVO中并添加到orderVOListorderVO.setOrderDishes(orderDishes);orderVOList.add(orderVO);}}return orderVOList;}/*** 根据订单id获取菜品信息字符串** param orders* return*/private String getOrderDishesStr(Orders orders) {// 查询订单菜品详情信息订单中的菜品和数量ListOrderDetail orderDetailList orderDetailMapper.getByOrderId(orders.getId());// 将每一条订单菜品信息拼接为字符串格式宫保鸡丁*3ListString orderDishList orderDetailList.stream().map(x - {String orderDish x.getName() * x.getNumber() ;;return orderDish;}).collect(Collectors.toList());// 将该订单对应的所有菜品信息拼接在一起return String.join(, orderDishList);}
Mapper层中
无新东西
订单搜索——功能测试 各个状态的订单数量统计——需求分析和设计
产品原型 接口设计 各个状态的订单数量统计——代码开发
Controller层中 /*** 各个状态的订单数量统计** return*/GetMapping(/statistics)ApiOperation(各个状态的订单数量统计)public ResultOrderStatisticsVO statistics() {OrderStatisticsVO orderStatisticsVO orderService.statistics();return Result.success(orderStatisticsVO);}
Service层中 /*** 各个状态的订单数量统计** return*/public OrderStatisticsVO statistics() {// 根据状态分别查询出待接单、待派送、派送中的订单数量Integer toBeConfirmed orderMapper.countStatus(Orders.TO_BE_CONFIRMED);Integer confirmed orderMapper.countStatus(Orders.CONFIRMED);Integer deliveryInProgress orderMapper.countStatus(Orders.DELIVERY_IN_PROGRESS);// 将查询出的数据封装到orderStatisticsVO中响应OrderStatisticsVO orderStatisticsVO new OrderStatisticsVO();orderStatisticsVO.setToBeConfirmed(toBeConfirmed);orderStatisticsVO.setConfirmed(confirmed);orderStatisticsVO.setDeliveryInProgress(deliveryInProgress);return orderStatisticsVO;}
Mapper层中 /*** 根据状态统计订单数量* param status*/Select(select count(id) from orders where status #{status})Integer countStatus(Integer status);
各个状态的订单数量统计——功能测试
成功显示 查询订单详情——需求分析和设计
产品原型 业务规则 订单详情页面需要展示订单基本信息状态、订单号、下单时间、收货人、电话、收货地址、金额等 订单详情页面需要展示订单明细数据商品名称、数量、单价
接口设计 查询订单详情——代码开发
Controller中 /*** 订单详情** param id* return*/GetMapping(/details/{id})ApiOperation(查询订单详情)public ResultOrderVO details(PathVariable(id) Long id) {OrderVO orderVO orderService.details(id);return Result.success(orderVO);}
Service中
前面写好了已经
Mapper中
前面写好了已经
查询订单详情——功能测试 接单——需求分析与设计
产品原型 业务规则 商家接单其实就是将订单的状态修改为“已接单”
接口设计 接单——代码开发
Controller中 /*** 接单** return*/PutMapping(/confirm)ApiOperation(接单)public Result confirm(RequestBody OrdersConfirmDTO ordersConfirmDTO) {orderService.confirm(ordersConfirmDTO);return Result.success();}
Service中 /*** 接单** param ordersConfirmDTO*/public void confirm(OrdersConfirmDTO ordersConfirmDTO) {Orders orders Orders.builder().id(ordersConfirmDTO.getId()).status(Orders.CONFIRMED).build();orderMapper.update(orders);}
Mapper中
前面已经写好了
接单——功能测试
接单后成功修改状态。 拒单——需求分析和设计
产品原型 业务规则 商家拒单其实就是将订单状态修改为“已取消” 只有订单处于“待接单”状态时可以执行拒单操作 商家拒单时需要指定拒单原因 商家拒单时如果用户已经完成了支付需要为用户退款
接口设计 拒单——代码开发
Controller中 /*** 拒单** return*/PutMapping(/rejection)ApiOperation(拒单)public Result rejection(RequestBody OrdersRejectionDTO ordersRejectionDTO) throws Exception {orderService.rejection(ordersRejectionDTO);return Result.success();}
Service中 /*** 拒单** param ordersRejectionDTO*/public void rejection(OrdersRejectionDTO ordersRejectionDTO) throws Exception {// 根据id查询订单Orders ordersDB orderMapper.getById(ordersRejectionDTO.getId());// 订单只有存在且状态为2待接单才可以拒单if (ordersDB null || !ordersDB.getStatus().equals(Orders.TO_BE_CONFIRMED)) {throw new OrderBusinessException(MessageConstant.ORDER_STATUS_ERROR);}//支付状态Integer payStatus ordersDB.getPayStatus();if (payStatus Orders.PAID) {//用户已支付需要退款String refund weChatPayUtil.refund(ordersDB.getNumber(),ordersDB.getNumber(),new BigDecimal(0.01),new BigDecimal(0.01));log.info(申请退款{}, refund);}// 拒单需要退款根据订单id更新订单状态、拒单原因、取消时间Orders orders new Orders();orders.setId(ordersDB.getId());orders.setStatus(Orders.CANCELLED);orders.setRejectionReason(ordersRejectionDTO.getRejectionReason());orders.setCancelTime(LocalDateTime.now());orderMapper.update(orders);}
Mapper中
前面写好的直接使用
拒单——功能测试
成功拒单 取消订单——需求分析和设计
产品原型 业务规则 取消订单其实就是将订单状态修改为“已取消” 商家取消订单时需要指定取消原因 商家取消订单时如果用户已经完成了支付需要为用户退款
接口设计 取消订单——代码开发
Controller中 /*** 取消订单** return*/PutMapping(/cancel)ApiOperation(取消订单)public Result cancel(RequestBody OrdersCancelDTO ordersCancelDTO) throws Exception {orderService.cancel(ordersCancelDTO);return Result.success();}
Service中 /*** 取消订单** param ordersCancelDTO*/public void cancel(OrdersCancelDTO ordersCancelDTO) throws Exception {// 根据id查询订单Orders ordersDB orderMapper.getById(ordersCancelDTO.getId());//支付状态Integer payStatus ordersDB.getPayStatus();if (payStatus 1) {//用户已支付需要退款String refund weChatPayUtil.refund(ordersDB.getNumber(),ordersDB.getNumber(),new BigDecimal(0.01),new BigDecimal(0.01));log.info(申请退款{}, refund);}// 管理端取消订单需要退款根据订单id更新订单状态、取消原因、取消时间Orders orders new Orders();orders.setId(ordersCancelDTO.getId());orders.setStatus(Orders.CANCELLED);orders.setCancelReason(ordersCancelDTO.getCancelReason());orders.setCancelTime(LocalDateTime.now());orderMapper.update(orders);}
Mapper中
用前面写好的的实现
取消订单——功能测试
成功取消
派送订单——需求分析和设计
产品原型 业务规则 派送订单其实就是将订单状态修改为“派送中” 只有状态为“待派送”的订单可以执行派送订单操作
接口设计 派送订单——代码开发
Controller中 /*** 派送订单** return*/PutMapping(/delivery/{id})ApiOperation(派送订单)public Result delivery(PathVariable(id) Long id) {orderService.delivery(id);return Result.success();}
Service中 /*** 派送订单** param id*/public void delivery(Long id) {// 根据id查询订单Orders ordersDB orderMapper.getById(id);// 校验订单是否存在并且状态为3if (ordersDB null || !ordersDB.getStatus().equals(Orders.CONFIRMED)) {throw new OrderBusinessException(MessageConstant.ORDER_STATUS_ERROR);}Orders orders new Orders();orders.setId(ordersDB.getId());// 更新订单状态,状态转为派送中orders.setStatus(Orders.DELIVERY_IN_PROGRESS);orderMapper.update(orders);}
Mapper中
使用前面写好的
派送订单——功能测试 完成订单——需求分析和设计
产品原型 业务规则 完成订单其实就是将订单状态修改为“已完成” 只有状态为“派送中”的订单可以执行订单完成操作
接口设计 完成订单——代码开发
Controller中 /*** 完成订单** return*/PutMapping(/complete/{id})ApiOperation(完成订单)public Result complete(PathVariable(id) Long id) {orderService.complete(id);return Result.success();}
Service中 /*** 完成订单** param id*/public void complete(Long id) {// 根据id查询订单Orders ordersDB orderMapper.getById(id);// 校验订单是否存在并且状态为4if (ordersDB null || !ordersDB.getStatus().equals(Orders.DELIVERY_IN_PROGRESS)) {throw new OrderBusinessException(MessageConstant.ORDER_STATUS_ERROR);}Orders orders new Orders();orders.setId(ordersDB.getId());// 更新订单状态,状态转为完成orders.setStatus(Orders.COMPLETED);orders.setDeliveryTime(LocalDateTime.now());orderMapper.update(orders);}
Mapper中
使用前面写好的
完成订单——功能测试 校验收货地址是否超出配送范围
1. 环境准备
注册账号百度地图开放平台 | 百度地图API SDK | 地图开发
登录百度地图开放平台百度地图开放平台 | 百度地图API SDK | 地图开发
进入控制台创建应用获取AK 相关接口:
逆地理编码 gc | 百度地图API SDK
webapi | 百度地图API SDK 2. 代码开发
2.1 application.yml
配置外卖商家店铺地址和百度地图的AK 2.2 OrderServiceImpl
改造OrderServiceImpl注入上面的配置项 Value(${sky.shop.address})private String shopAddress;Value(${sky.baidu.ak})private String ak;
在OrderServiceImpl中提供校验方法
/*** 检查客户的收货地址是否超出配送范围* param address*/private void checkOutOfRange(String address) {Map map new HashMap();map.put(address,shopAddress);map.put(output,json);map.put(ak,ak);//获取店铺的经纬度坐标String shopCoordinate HttpClientUtil.doGet(https://api.map.baidu.com/geocoding/v3, map);JSONObject jsonObject JSON.parseObject(shopCoordinate);if(!jsonObject.getString(status).equals(0)){throw new OrderBusinessException(店铺地址解析失败);}//数据解析JSONObject location jsonObject.getJSONObject(result).getJSONObject(location);String lat location.getString(lat);String lng location.getString(lng);//店铺经纬度坐标String shopLngLat lat , lng;map.put(address,address);//获取用户收货地址的经纬度坐标String userCoordinate HttpClientUtil.doGet(https://api.map.baidu.com/geocoding/v3, map);jsonObject JSON.parseObject(userCoordinate);if(!jsonObject.getString(status).equals(0)){throw new OrderBusinessException(收货地址解析失败);}//数据解析location jsonObject.getJSONObject(result).getJSONObject(location);lat location.getString(lat);lng location.getString(lng);//用户收货地址经纬度坐标String userLngLat lat , lng;map.put(origin,shopLngLat);map.put(destination,userLngLat);map.put(steps_info,0);//路线规划String json HttpClientUtil.doGet(https://api.map.baidu.com/directionlite/v1/driving, map);jsonObject JSON.parseObject(json);if(!jsonObject.getString(status).equals(0)){throw new OrderBusinessException(配送路线规划失败);}//数据解析JSONObject result jsonObject.getJSONObject(result);JSONArray jsonArray (JSONArray) result.get(routes);Integer distance (Integer) ((JSONObject) jsonArray.get(0)).get(distance);if(distance 5000){//配送距离超过5000米throw new OrderBusinessException(超出配送范围);}}
在OrderServiceImpl的submitOrder方法中调用上面的校验方法 功能测试
距离过大直接报错并且会卡在这个页面 距离近时可以直接进入支付页面