|  |  |  | 
|---|
|  |  |  | package com.zy.common.utils; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import com.baomidou.mybatisplus.mapper.EntityWrapper; | 
|---|
|  |  |  | import com.zy.asrs.entity.BasShuttle; | 
|---|
|  |  |  | import com.zy.asrs.service.BasShuttleService; | 
|---|
|  |  |  | import com.core.common.SpringUtils; | 
|---|
|  |  |  | import com.core.exception.CoolException; | 
|---|
|  |  |  | import com.zy.asrs.entity.DeviceConfig; | 
|---|
|  |  |  | import com.zy.asrs.service.DeviceConfigService; | 
|---|
|  |  |  | import com.zy.asrs.utils.Utils; | 
|---|
|  |  |  | import com.zy.common.model.MapNode; | 
|---|
|  |  |  | import com.zy.common.model.NavigateNode; | 
|---|
|  |  |  | import com.zy.common.model.enums.NavigationMapType; | 
|---|
|  |  |  | import com.zy.core.News; | 
|---|
|  |  |  | import com.zy.core.cache.SlaveConnection; | 
|---|
|  |  |  | import com.zy.core.dispatcher.ShuttleDispatchUtils; | 
|---|
|  |  |  | import com.zy.core.enums.MapNodeType; | 
|---|
|  |  |  | import com.zy.core.enums.RedisKeyType; | 
|---|
|  |  |  | import com.zy.core.enums.ShuttleRunDirection; | 
|---|
|  |  |  | import com.zy.core.enums.SlaveType; | 
|---|
|  |  |  | import com.zy.core.model.command.ShuttleAssignCommand; | 
|---|
|  |  |  | import com.zy.core.model.command.ShuttleCommand; | 
|---|
|  |  |  | import com.zy.core.model.protocol.ShuttleProtocol; | 
|---|
|  |  |  | import com.zy.core.thread.ShuttleThread; | 
|---|
|  |  |  | import com.zy.system.entity.Config; | 
|---|
|  |  |  | import com.zy.system.service.ConfigService; | 
|---|
|  |  |  | import org.springframework.beans.factory.annotation.Autowired; | 
|---|
|  |  |  | import org.springframework.stereotype.Component; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import java.util.ArrayList; | 
|---|
|  |  |  | import java.util.List; | 
|---|
|  |  |  | import java.util.Optional; | 
|---|
|  |  |  | import java.util.*; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 四向车操作工具类 | 
|---|
|  |  |  | 
|---|
|  |  |  | public class ShuttleOperaUtils { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private BasShuttleService basShuttleService; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private NavigateUtils navigateUtils; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private NavigateMapUtils navigateMapUtils; | 
|---|
|  |  |  | private NavigateMapData navigateMapData; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ConfigService configService; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private ShuttleDispatchUtils shuttleDispatchUtils; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private DeviceConfigService deviceConfigService; | 
|---|
|  |  |  | @Autowired | 
|---|
|  |  |  | private RedisUtil redisUtil; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> getStartToTargetCommands(String startLocNo, String endLocNo, Integer mapType, ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread) { | 
|---|
|  |  |  | return getStartToTargetCommands(startLocNo, endLocNo, mapType, null, assignCommand, shuttleThread); | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> getStartToTargetCommands(String startLocNo, String endLocNo, List<NavigationMapType> mapTypes, ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread) { | 
|---|
|  |  |  | return getStartToTargetCommands(startLocNo, endLocNo, mapTypes, null, assignCommand, shuttleThread, "move"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> getStartToTargetCommands(String startLocNo, String endLocNo, Integer mapType, List<int[]> whites, ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread) { | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> getStartToTargetCommands(String startLocNo, String endLocNo, List<NavigationMapType> mapTypes, ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread, String moveType) { | 
|---|
|  |  |  | return getStartToTargetCommands(startLocNo, endLocNo, mapTypes, null, assignCommand, shuttleThread, moveType); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> getStartToTargetCommands(String startLocNo, String endLocNo, List<NavigationMapType> mapTypes, List<int[]> whites, ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread, String moveType) { | 
|---|
|  |  |  | ShuttleProtocol shuttleProtocol = shuttleThread.getStatus(); | 
|---|
|  |  |  | if (shuttleProtocol == null) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Integer shuttleNo = shuttleProtocol.getShuttleNo(); | 
|---|
|  |  |  | //获取小车移动速度 | 
|---|
|  |  |  | Integer runSpeed = Optional.ofNullable(basShuttleService.selectOne(new EntityWrapper<BasShuttle>().eq("shuttle_no", shuttleNo)).getRunSpeed()).orElse(1000); | 
|---|
|  |  |  | List<NavigateNode> nodeList = navigateUtils.calc(startLocNo, endLocNo, mapType, Utils.getShuttlePoints(shuttleNo, Utils.getLev(startLocNo)), null); | 
|---|
|  |  |  | Integer runSpeed = 1000; | 
|---|
|  |  |  | Object speedObj = redisUtil.get(RedisKeyType.SHUTTLE_SPEED_MAP.key); | 
|---|
|  |  |  | if (speedObj != null) { | 
|---|
|  |  |  | HashMap<Integer, Integer> shuttleSpeedMap = (HashMap<Integer, Integer>) speedObj; | 
|---|
|  |  |  | Integer speed = shuttleSpeedMap.get(shuttleNo); | 
|---|
|  |  |  | if (speed != null) { | 
|---|
|  |  |  | runSpeed = speed; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Object systemConfigMapObj = redisUtil.get(RedisKeyType.SYSTEM_CONFIG_MAP.key); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //小车(x,y)命令运行方向颠倒 | 
|---|
|  |  |  | boolean shuttleDirectionReverse = false; | 
|---|
|  |  |  | if (systemConfigMapObj != null) { | 
|---|
|  |  |  | HashMap<String, String> systemConfigMap = (HashMap<String, String>) systemConfigMapObj; | 
|---|
|  |  |  | if (systemConfigMap.get("shuttleDirectionReverse").equals("Y")) { | 
|---|
|  |  |  | shuttleDirectionReverse = true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | long calcStartTime = System.currentTimeMillis(); | 
|---|
|  |  |  | List<NavigateNode> nodeList = navigateUtils.calc(startLocNo, endLocNo, mapTypes, Utils.getShuttlePoints(shuttleNo, Utils.getLev(startLocNo)), whites); | 
|---|
|  |  |  | if (nodeList == null) { | 
|---|
|  |  |  | News.error("{} dash {} can't find navigate path!", startLocNo, endLocNo); | 
|---|
|  |  |  | shuttleThread.offerSystemMsg("{} dash {} can't find navigate path!", startLocNo, endLocNo); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | News.info("[RCS Debug] Calc path time:{}", (System.currentTimeMillis() - calcStartTime)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<NavigateNode> allNode = new ArrayList<>(); | 
|---|
|  |  |  | List<NavigateNode> lockNode = new ArrayList<>(); | 
|---|
|  |  |  | for (NavigateNode node : nodeList) { | 
|---|
|  |  |  | allNode.add(node.clone()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (whites != null) { | 
|---|
|  |  |  | boolean flag = false; | 
|---|
|  |  |  | for (int[] white : whites) { | 
|---|
|  |  |  | if (white[0] == node.getX() && white[1] == node.getY()) { | 
|---|
|  |  |  | flag = true; | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (flag) {//白名单跳过锁定 | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | lockNode.add(node.clone()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | long startTime = System.currentTimeMillis(); | 
|---|
|  |  |  | List<ShuttleCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  | //获取分段路径 | 
|---|
|  |  |  | ArrayList<ArrayList<NavigateNode>> data = navigateUtils.getSectionPath(nodeList); | 
|---|
|  |  |  | List<List<NavigateNode>> data = navigateUtils.getSectionPath(nodeList); | 
|---|
|  |  |  | long endTime = System.currentTimeMillis(); | 
|---|
|  |  |  | News.info("[RCS Debug] getSection path time:{}", (endTime - startTime)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | long startGetSectionCommandTime = System.currentTimeMillis(); | 
|---|
|  |  |  | //将每一段路径分成command指令 | 
|---|
|  |  |  | for (ArrayList<NavigateNode> nodes : data) { | 
|---|
|  |  |  | for (int i = 0; i < data.size(); i++) { | 
|---|
|  |  |  | long startGetStartAndEndPathTime = System.currentTimeMillis(); | 
|---|
|  |  |  | List<NavigateNode> nodes = data.get(i); | 
|---|
|  |  |  | //开始路径 | 
|---|
|  |  |  | NavigateNode startPath = nodes.get(0); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //目标路径 | 
|---|
|  |  |  | NavigateNode endPath = nodes.get(nodes.size() - 1); | 
|---|
|  |  |  | News.info("[RCS Debug] getStartAndEndPath idx:{} time:{}", i, (System.currentTimeMillis() - startGetStartAndEndPathTime)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | long startAllDistanceTime = System.currentTimeMillis(); | 
|---|
|  |  |  | Integer allDistance = navigateUtils.getCurrentPathAllDistance(nodes);//计算当前路径行走总距离 | 
|---|
|  |  |  | News.info("[RCS Debug] getAllDistance idx:{} time:{}", i, (System.currentTimeMillis() - startAllDistanceTime)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | long startGetPosition = System.currentTimeMillis(); | 
|---|
|  |  |  | //通过xy坐标小车二维码 | 
|---|
|  |  |  | String startCodeNum = NavigatePositionConvert.xyToPosition(startPath.getX(), startPath.getY(), startPath.getZ()); | 
|---|
|  |  |  | //通过xy坐标小车二维码 | 
|---|
|  |  |  | String distCodeNum = NavigatePositionConvert.xyToPosition(endPath.getX(), endPath.getY(), endPath.getZ()); | 
|---|
|  |  |  | News.info("[RCS Debug] xyToPosition idx:{} time:{}", i, (System.currentTimeMillis() - startGetPosition)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | long startGetMoveCommandTime = System.currentTimeMillis(); | 
|---|
|  |  |  | //获取移动命令 | 
|---|
|  |  |  | ShuttleCommand command = shuttleThread.getMoveCommand(assignCommand.getDeviceTaskNo(), startCodeNum, distCodeNum, allDistance, ShuttleRunDirection.get(startPath.getDirection()).id.intValue(), runSpeed, nodes); | 
|---|
|  |  |  | ShuttleCommand command = shuttleThread.getMoveCommand(assignCommand.getDeviceTaskNo(), startCodeNum, distCodeNum, allDistance, ShuttleRunDirection.get(startPath.getDirection()).id.intValue(), runSpeed, nodes, shuttleDirectionReverse); | 
|---|
|  |  |  | News.info("[RCS Debug] getMoveCommand idx:{} time:{}", i, (System.currentTimeMillis() - startGetMoveCommandTime)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (i + 1 == data.size()) { | 
|---|
|  |  |  | long startGetInOutLiftTime = System.currentTimeMillis(); | 
|---|
|  |  |  | if (moveType.equals("inLift")) { | 
|---|
|  |  |  | command = shuttleThread.getMoveLiftCommand(assignCommand.getDeviceTaskNo(), startCodeNum, distCodeNum, allDistance, ShuttleRunDirection.get(startPath.getDirection()).id.intValue(), runSpeed, nodes, true, shuttleDirectionReverse); | 
|---|
|  |  |  | }else if (moveType.equals("outLift")) { | 
|---|
|  |  |  | command = shuttleThread.getMoveLiftCommand(assignCommand.getDeviceTaskNo(), startCodeNum, distCodeNum, allDistance, ShuttleRunDirection.get(startPath.getDirection()).id.intValue(), runSpeed, nodes, false, shuttleDirectionReverse); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | News.info("[RCS Debug] getInOutLiftTime idx:{} time:{}", i, (System.currentTimeMillis() - startGetInOutLiftTime)); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | command.setNodes(nodes);//将行走节点添加到每一步命令中 | 
|---|
|  |  |  | commands.add(command); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | News.info("[RCS Debug] getSectionCommand path time:{}", (System.currentTimeMillis() - startGetSectionCommandTime)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | assignCommand.setNodes(allNode);//当前任务所占用的节点list | 
|---|
|  |  |  | assignCommand.setMapTypes(mapTypes); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return commands; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> shuttleInOutLiftCommand(String startLocNo, String endLocNo, Integer mapType, ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread) { | 
|---|
|  |  |  | NavigateNode startNode = NavigatePositionConvert.locNoToNode(startLocNo); | 
|---|
|  |  |  | NavigateNode endNode = NavigatePositionConvert.locNoToNode(endLocNo); | 
|---|
|  |  |  | List<NavigateNode> unlockPath = new ArrayList<>(); | 
|---|
|  |  |  | unlockPath.add(startNode); | 
|---|
|  |  |  | unlockPath.add(endNode); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | ShuttleProtocol shuttleProtocol = shuttleThread.getStatus(); | 
|---|
|  |  |  | if (shuttleProtocol == null) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | //小车移动连续下发指令 | 
|---|
|  |  |  | boolean shuttleMoveCommandsContinuously = false; | 
|---|
|  |  |  | if (systemConfigMapObj != null) { | 
|---|
|  |  |  | HashMap<String, String> systemConfigMap = (HashMap<String, String>) systemConfigMapObj; | 
|---|
|  |  |  | String shuttleMoveCommandsContinuouslyConfig = systemConfigMap.get("shuttleMoveCommandsContinuously"); | 
|---|
|  |  |  | if (shuttleMoveCommandsContinuouslyConfig != null && shuttleMoveCommandsContinuouslyConfig.equals("Y")) { | 
|---|
|  |  |  | shuttleMoveCommandsContinuously = true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | assignCommand.setShuttleMoveCommandsContinuously(shuttleMoveCommandsContinuously); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Integer shuttleNo = shuttleProtocol.getShuttleNo(); | 
|---|
|  |  |  | //所使用的路径进行锁定/解锁 | 
|---|
|  |  |  | boolean lockResult = navigateMapUtils.writeNavigateNodeToRedisMap(Utils.getLev(endLocNo), shuttleProtocol.getShuttleNo(), unlockPath, false);//所使用的路径进行解锁 | 
|---|
|  |  |  | if (!lockResult) { | 
|---|
|  |  |  | News.error("{} dash {} can't find unlock path!", startLocNo, endLocNo); | 
|---|
|  |  |  | return null;//解锁失败 | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //获取小车移动速度 | 
|---|
|  |  |  | Integer runSpeed = Optional.ofNullable(basShuttleService.selectOne(new EntityWrapper<BasShuttle>().eq("shuttle_no", shuttleNo)).getRunSpeed()).orElse(1000); | 
|---|
|  |  |  | List<NavigateNode> nodeList = navigateUtils.calc(startLocNo, endLocNo, mapType, Utils.getShuttlePoints(shuttleNo, Utils.getLev(startLocNo)), null); | 
|---|
|  |  |  | if (nodeList == null) { | 
|---|
|  |  |  | News.error("{} dash {} can't find navigate path!", startLocNo, endLocNo); | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<NavigateNode> allNode = new ArrayList<>(); | 
|---|
|  |  |  | for (NavigateNode node : nodeList) { | 
|---|
|  |  |  | allNode.add(node.clone()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<ShuttleCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  | //获取分段路径 | 
|---|
|  |  |  | ArrayList<ArrayList<NavigateNode>> data = navigateUtils.getSectionPath(nodeList); | 
|---|
|  |  |  | //将每一段路径分成command指令 | 
|---|
|  |  |  | for (ArrayList<NavigateNode> nodes : data) { | 
|---|
|  |  |  | //开始路径 | 
|---|
|  |  |  | NavigateNode startPath = nodes.get(0); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //目标路径 | 
|---|
|  |  |  | NavigateNode endPath = nodes.get(nodes.size() - 1); | 
|---|
|  |  |  | Integer allDistance = navigateUtils.getCurrentPathAllDistance(nodes);//计算当前路径行走总距离 | 
|---|
|  |  |  | //通过xy坐标小车二维码 | 
|---|
|  |  |  | String startCodeNum = NavigatePositionConvert.xyToPosition(startPath.getX(), startPath.getY(), startPath.getZ()); | 
|---|
|  |  |  | //通过xy坐标小车二维码 | 
|---|
|  |  |  | String distCodeNum = NavigatePositionConvert.xyToPosition(endPath.getX(), endPath.getY(), endPath.getZ()); | 
|---|
|  |  |  | //获取移动命令 | 
|---|
|  |  |  | ShuttleCommand command = shuttleThread.getMoveCommand(assignCommand.getDeviceTaskNo(), startCodeNum, distCodeNum, allDistance, ShuttleRunDirection.get(startPath.getDirection()).id.intValue(), runSpeed, nodes); | 
|---|
|  |  |  | command.setNodes(nodes);//将行走节点添加到每一步命令中 | 
|---|
|  |  |  | commands.add(command); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | assignCommand.setNodes(allNode);//当前任务所占用的节点list | 
|---|
|  |  |  |  | 
|---|
|  |  |  | News.info("{}任务,{}小车,{} - {} 路径命令包计算成功,耗时:{}ms", assignCommand.getTaskNo(), shuttleProtocol.getShuttleNo(), startLocNo, endLocNo, System.currentTimeMillis() - startTime); | 
|---|
|  |  |  | return commands; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 获取充电命令 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> shuttleChargeCommand(ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread, Boolean openCharge) { | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> getShuttleChargeCommand(ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread, Boolean openCharge) { | 
|---|
|  |  |  | List<ShuttleCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //获取充电命令 | 
|---|
|  |  |  | ShuttleCommand command = shuttleThread.getChargeCommand(assignCommand.getDeviceTaskNo(), openCharge); | 
|---|
|  |  |  | commands.add(command); | 
|---|
|  |  |  | return commands; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 获取顶升命令 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public synchronized List<ShuttleCommand> getShuttleLiftCommand(ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread, Boolean lift) { | 
|---|
|  |  |  | List<ShuttleCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //获取顶升命令 | 
|---|
|  |  |  | ShuttleCommand command = shuttleThread.getLiftCommand(assignCommand.getDeviceTaskNo(), lift); | 
|---|
|  |  |  | commands.add(command); | 
|---|
|  |  |  | return commands; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 检测障碍物车 | 
|---|
|  |  |  | * @return 0:无障碍 1:有障碍调度成功 2:有障碍调度失败 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public synchronized int checkObstacle(String locNo, List<Integer> whiteShuttles, List<NavigateNode> whiteNodes) { | 
|---|
|  |  |  | int innerCircle = 0; | 
|---|
|  |  |  | int outerCircle = 3; | 
|---|
|  |  |  | Config avoidInnerCircleConfig = configService.selectOne(new EntityWrapper<Config>().eq("code", "avoidInnerCircle")); | 
|---|
|  |  |  | if (avoidInnerCircleConfig != null) { | 
|---|
|  |  |  | innerCircle = Integer.parseInt(avoidInnerCircleConfig.getValue()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Config avoidOuterCircleConfig = configService.selectOne(new EntityWrapper<Config>().eq("code", "avoidOuterCircle")); | 
|---|
|  |  |  | if (avoidOuterCircleConfig != null) { | 
|---|
|  |  |  | outerCircle = Integer.parseInt(avoidOuterCircleConfig.getValue()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | ArrayList<int[]> whiteShuttlePointList = new ArrayList<>(); | 
|---|
|  |  |  | for (Integer shuttle : whiteShuttles) { | 
|---|
|  |  |  | //获取四向穿梭车线程 | 
|---|
|  |  |  | ShuttleThread shuttleThread = (ShuttleThread) SlaveConnection.get(SlaveType.Shuttle, shuttle); | 
|---|
|  |  |  | if (shuttleThread == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | ShuttleProtocol shuttleProtocol = shuttleThread.getStatus(); | 
|---|
|  |  |  | if (shuttleProtocol == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String currentLocNo = shuttleProtocol.getCurrentLocNo(); | 
|---|
|  |  |  | int[] shuttleArr = NavigatePositionConvert.positionToXY(currentLocNo); | 
|---|
|  |  |  | whiteShuttlePointList.add(shuttleArr); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //获取内圈节点 | 
|---|
|  |  |  | List<NavigateNode> innerNodes = getInnerNodes(locNo, innerCircle, whiteShuttlePointList); | 
|---|
|  |  |  | List<Integer> nodesCar = findNodesCar(innerNodes); | 
|---|
|  |  |  | if (nodesCar.isEmpty()) { | 
|---|
|  |  |  | return 0;//内圈中无车 | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //获取外圈节点 | 
|---|
|  |  |  | List<NavigateNode> outerNodes = getOuterNodes(locNo, outerCircle, whiteShuttlePointList, innerNodes, whiteNodes); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //将内圈节点中障碍小车调离 | 
|---|
|  |  |  | for (Integer shuttleNo : nodesCar) { | 
|---|
|  |  |  | ShuttleThread shuttleThread = (ShuttleThread) SlaveConnection.get(SlaveType.Shuttle, shuttleNo); | 
|---|
|  |  |  | if (shuttleThread == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | ShuttleProtocol shuttleProtocol = shuttleThread.getStatus(); | 
|---|
|  |  |  | if (shuttleProtocol == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (!shuttleThread.isIdle()) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | String targetLocNo = null; | 
|---|
|  |  |  | for (NavigateNode node : outerNodes) { | 
|---|
|  |  |  | String dispatchLocNo = NavigatePositionConvert.nodeToLocNo(node); | 
|---|
|  |  |  | //获取内圈节点 | 
|---|
|  |  |  | List<NavigateNode> avoidInnerNodes = getInnerNodes(dispatchLocNo, innerCircle, new ArrayList<>()); | 
|---|
|  |  |  | //计算内圈是否有小车 | 
|---|
|  |  |  | List<Integer> avoidNodesCar = findNodesCar(avoidInnerNodes); | 
|---|
|  |  |  | if (!avoidNodesCar.isEmpty()) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | targetLocNo = dispatchLocNo; | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (targetLocNo == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | boolean dispatched = shuttleDispatchUtils.dispatchShuttle(null, targetLocNo, shuttleNo); | 
|---|
|  |  |  | return dispatched ? 1 : 2; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return 2;//内圈中有车 | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private HashMap<String, Integer> findCarMap() { | 
|---|
|  |  |  | HashMap<String, Integer> carMap = new HashMap<>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<DeviceConfig> shuttleList = deviceConfigService.selectList(new EntityWrapper<DeviceConfig>() | 
|---|
|  |  |  | .eq("device_type", String.valueOf(SlaveType.Shuttle))); | 
|---|
|  |  |  | for (DeviceConfig device : shuttleList) { | 
|---|
|  |  |  | ShuttleThread shuttleThread = (ShuttleThread) SlaveConnection.get(SlaveType.Shuttle, device.getDeviceNo()); | 
|---|
|  |  |  | if (shuttleThread == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | ShuttleProtocol shuttleProtocol = shuttleThread.getStatus(); | 
|---|
|  |  |  | if (shuttleProtocol == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | String currentLocNo = shuttleProtocol.getCurrentLocNo(); | 
|---|
|  |  |  | if (currentLocNo == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | carMap.put(currentLocNo, device.getDeviceNo()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return carMap; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //搜索节点内的小车编号 | 
|---|
|  |  |  | private List<Integer> findNodesCar(List<NavigateNode> nodes) { | 
|---|
|  |  |  | List<Integer> list = new ArrayList<>(); | 
|---|
|  |  |  | HashMap<String, Integer> carMap = findCarMap(); | 
|---|
|  |  |  | for (NavigateNode node : nodes) { | 
|---|
|  |  |  | String locNo = NavigatePositionConvert.nodeToLocNo(node); | 
|---|
|  |  |  | if (carMap.containsKey(locNo)) { | 
|---|
|  |  |  | list.add(carMap.get(locNo)); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return list; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private List<NavigateNode> getOuterNodes(String locNo, int outerCircle, List<int[]> whiteShuttlePointList, List<NavigateNode> innerNodes, List<NavigateNode> whiteNodes) { | 
|---|
|  |  |  | List<NavigateNode> outerNodes = new ArrayList<>(); | 
|---|
|  |  |  | List<NavigateNode> outerNodesTmp = new ArrayList<>(); | 
|---|
|  |  |  | int lev = Utils.getLev(locNo); | 
|---|
|  |  |  | int[] pointArr = NavigatePositionConvert.positionToXY(locNo); | 
|---|
|  |  |  | NavigateNode currentNode = new NavigateNode(pointArr[0], pointArr[1]); | 
|---|
|  |  |  | currentNode.setZ(lev); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<List<MapNode>> lists = navigateMapData.getJsonData(lev, NavigationMapType.getMapTypes(NavigationMapType.NONE), null, null);//获取完整地图 | 
|---|
|  |  |  | int[][] map = navigateMapData.parseJsonDataArr(lists); | 
|---|
|  |  |  | int nodeValue = map[pointArr[0]][pointArr[1]]; | 
|---|
|  |  |  | currentNode.setNodeValue(nodeValue); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | for (int i = 0; i < outerCircle; i++) { | 
|---|
|  |  |  | int idx = i + 1; | 
|---|
|  |  |  | List<NavigateNode> list = extend_outer_nodes(map, currentNode, idx); | 
|---|
|  |  |  | if (list.isEmpty()) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | outerNodesTmp.addAll(list); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | for (NavigateNode node : outerNodesTmp) { | 
|---|
|  |  |  | boolean flag = false; | 
|---|
|  |  |  | for (NavigateNode innerNode : innerNodes) { | 
|---|
|  |  |  | if (node.getX() == innerNode.getX() && node.getY() == innerNode.getY()) { | 
|---|
|  |  |  | flag = true; | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (flag) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | for (NavigateNode whiteNode : whiteNodes) { | 
|---|
|  |  |  | if (node.getX() == whiteNode.getX() && node.getY() == whiteNode.getY()) { | 
|---|
|  |  |  | flag = true; | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (flag) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | outerNodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return outerNodes; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private List<NavigateNode> getInnerNodes(String locNo, int innerCircle, List<int[]> whiteShuttlePointList) { | 
|---|
|  |  |  | List<NavigateNode> innerNodes = new ArrayList<>(); | 
|---|
|  |  |  | int lev = Utils.getLev(locNo); | 
|---|
|  |  |  | int[] pointArr = NavigatePositionConvert.positionToXY(locNo); | 
|---|
|  |  |  | NavigateNode currentNode = new NavigateNode(pointArr[0], pointArr[1]); | 
|---|
|  |  |  | currentNode.setZ(lev); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | boolean addCurrentNode = true; | 
|---|
|  |  |  | for (int[] shuttlePoint : whiteShuttlePointList) { | 
|---|
|  |  |  | if(currentNode.getX() == shuttlePoint[0] && currentNode.getY() == shuttlePoint[1]) { | 
|---|
|  |  |  | addCurrentNode = false; | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (addCurrentNode) { | 
|---|
|  |  |  | innerNodes.add(currentNode); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<List<MapNode>> lists = navigateMapData.getJsonData(lev, NavigationMapType.getMapTypes(NavigationMapType.NONE), null, null);//获取完整地图 | 
|---|
|  |  |  | int[][] map = navigateMapData.parseJsonDataArr(lists); | 
|---|
|  |  |  | int nodeValue = map[pointArr[0]][pointArr[1]]; | 
|---|
|  |  |  | currentNode.setNodeValue(nodeValue); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | for (int i = 0; i < innerCircle; i++) { | 
|---|
|  |  |  | int idx = i + 1; | 
|---|
|  |  |  | List<NavigateNode> list = extend_inner_nodes(map, currentNode, idx); | 
|---|
|  |  |  | if (list.isEmpty()) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | for (NavigateNode node : list) { | 
|---|
|  |  |  | boolean flag = false; | 
|---|
|  |  |  | for (int[] shuttlePoint : whiteShuttlePointList) { | 
|---|
|  |  |  | if(node.getX() == shuttlePoint[0] && node.getY() == shuttlePoint[1]) { | 
|---|
|  |  |  | flag = true; | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if(flag) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | innerNodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return innerNodes; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private List<NavigateNode> extend_inner_nodes(int[][] map, NavigateNode startNode, int innerCircleIdx) { | 
|---|
|  |  |  | //默认地图母轨方向x | 
|---|
|  |  |  | String mapDirection = "x"; | 
|---|
|  |  |  | ConfigService configService = SpringUtils.getBean(ConfigService.class); | 
|---|
|  |  |  | if (configService != null) { | 
|---|
|  |  |  | Config config = configService.selectOne(new EntityWrapper<Config>() | 
|---|
|  |  |  | .eq("code", "direction_map") | 
|---|
|  |  |  | .eq("status", 1)); | 
|---|
|  |  |  | if (config != null) { | 
|---|
|  |  |  | mapDirection = config.getValue(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | ArrayList<NavigateNode> nodes = new ArrayList<>(); | 
|---|
|  |  |  | int x = startNode.getX(); | 
|---|
|  |  |  | int y = startNode.getY(); | 
|---|
|  |  |  | int z = startNode.getZ(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (mapDirection.equals("x")) {//母轨x方向 | 
|---|
|  |  |  | if (map[x][y] == MapNodeType.MAIN_PATH.id) { | 
|---|
|  |  |  | //母轨才能进行上下移动 | 
|---|
|  |  |  | if (is_valid(map, x + innerCircleIdx, y)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x + innerCircleIdx, y); | 
|---|
|  |  |  | node.setNodeValue(map[x + innerCircleIdx][y]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (is_valid(map, x - innerCircleIdx, y)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x - innerCircleIdx, y); | 
|---|
|  |  |  | node.setNodeValue(map[x - innerCircleIdx][y]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (map[x][y] == MapNodeType.NORMAL_PATH.id || map[x][y] == MapNodeType.MAIN_PATH.id || map[x][y] == MapNodeType.CONVEYOR_CAR_GO.id || map[x][y] == MapNodeType.CHARGE.id || map[x][y] == MapNodeType.LIFT.id) { | 
|---|
|  |  |  | //子轨和母轨、小车可走输送线、充电桩、提升机才能进行左右移动 | 
|---|
|  |  |  | if (is_valid(map, x, y + innerCircleIdx)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x, y + innerCircleIdx); | 
|---|
|  |  |  | node.setNodeValue(map[x][y + innerCircleIdx]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (is_valid(map, x, y - innerCircleIdx)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x, y - innerCircleIdx); | 
|---|
|  |  |  | node.setNodeValue(map[x][y - innerCircleIdx]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else if (mapDirection.equals("y")) {//母轨y方向 | 
|---|
|  |  |  | if (map[x][y] == MapNodeType.MAIN_PATH.id) { | 
|---|
|  |  |  | //母轨才能进行左右移动 | 
|---|
|  |  |  | if (is_valid(map, x, y + innerCircleIdx)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x, y + innerCircleIdx); | 
|---|
|  |  |  | node.setNodeValue(map[x][y + innerCircleIdx]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (is_valid(map, x, y - innerCircleIdx)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x, y - innerCircleIdx); | 
|---|
|  |  |  | node.setNodeValue(map[x][y - innerCircleIdx]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (map[x][y] == MapNodeType.NORMAL_PATH.id || map[x][y] == MapNodeType.MAIN_PATH.id || map[x][y] == MapNodeType.CONVEYOR_CAR_GO.id || map[x][y] == MapNodeType.CHARGE.id || map[x][y] == MapNodeType.LIFT.id) { | 
|---|
|  |  |  | //子轨和母轨、小车可走输送线、充电桩、提升机才能进行上下移动 | 
|---|
|  |  |  | if (is_valid(map, x + innerCircleIdx, y)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x + innerCircleIdx, y); | 
|---|
|  |  |  | node.setNodeValue(map[x + innerCircleIdx][y]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | if (is_valid(map, x - innerCircleIdx, y)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x - innerCircleIdx, y); | 
|---|
|  |  |  | node.setNodeValue(map[x - innerCircleIdx][y]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | throw new CoolException("异常地图方向"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return nodes; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private List<NavigateNode> extend_outer_nodes(int[][] map, NavigateNode startNode, int innerCircleIdx) { | 
|---|
|  |  |  | ArrayList<NavigateNode> nodes = new ArrayList<>(); | 
|---|
|  |  |  | int x = startNode.getX(); | 
|---|
|  |  |  | int y = startNode.getY(); | 
|---|
|  |  |  | int z = startNode.getZ(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (is_valid(map, x + innerCircleIdx, y)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x + innerCircleIdx, y); | 
|---|
|  |  |  | node.setNodeValue(map[x + innerCircleIdx][y]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (is_valid(map, x - innerCircleIdx, y)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x - innerCircleIdx, y); | 
|---|
|  |  |  | node.setNodeValue(map[x - innerCircleIdx][y]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (is_valid(map, x, y + innerCircleIdx)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x, y + innerCircleIdx); | 
|---|
|  |  |  | node.setNodeValue(map[x][y + innerCircleIdx]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (is_valid(map, x, y - innerCircleIdx)) { | 
|---|
|  |  |  | NavigateNode node = new NavigateNode(x, y - innerCircleIdx); | 
|---|
|  |  |  | node.setNodeValue(map[x][y - innerCircleIdx]); | 
|---|
|  |  |  | node.setZ(z); | 
|---|
|  |  |  | if (node.getNodeValue().equals(startNode.getNodeValue())) { | 
|---|
|  |  |  | nodes.add(node); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return nodes; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private boolean is_valid(int[][] map, int x, int y) { | 
|---|
|  |  |  | if (x < 0 || x >= map.length | 
|---|
|  |  |  | || y < 0 || y >= map[1].length) { | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | // 如果结点的位置小于0,则不合法 | 
|---|
|  |  |  | if (map[x][y] < 0) { | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //以上情况都没有则合法 | 
|---|
|  |  |  | return true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //    private boolean checkSimilarityPath(Motion motion, ShuttleAssignCommand assignCommand) { | 
|---|
|  |  |  | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        return true; | 
|---|
|  |  |  | //    } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //    /** | 
|---|
|  |  |  | //     * 计算并获取小车从起点——中间点——目标点之间搬运货物动作命令 | 
|---|
|  |  |  | //     * @param shuttleNo 小车号 | 
|---|
|  |  |  | //     * @param wrkNo 工作号 | 
|---|
|  |  |  | //     * @param startPoint 起点(小车当前位置) | 
|---|
|  |  |  | //     * @param targetPoint 目标点(货物目标位置) | 
|---|
|  |  |  | //     */ | 
|---|
|  |  |  | //    public static synchronized ShuttleOperaResult getShuttleTransportCommands(Integer shuttleNo, Integer wrkNo, String startPoint, String targetPoint) { | 
|---|
|  |  |  | //        //行走路径 | 
|---|
|  |  |  | //        ArrayList<NavigateNode> nodes = new ArrayList<>(); | 
|---|
|  |  |  | //        //命令集合 | 
|---|
|  |  |  | //        ArrayList<NyShuttleHttpCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        //计算起点到目标点命令 | 
|---|
|  |  |  | //        ShuttleOperaResult result = getStartToTargetCommands(shuttleNo, wrkNo, startPoint, targetPoint, NavigationMapType.DFX.id); | 
|---|
|  |  |  | //        if (result == null) { | 
|---|
|  |  |  | //            //计算结果必须不为空,否则计算失败 | 
|---|
|  |  |  | //            return null; | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | //        nodes.addAll(result.getNodes()); | 
|---|
|  |  |  | //        //起点位置下发一条顶升命令将货物进行搬运 | 
|---|
|  |  |  | //        commands.add(NyHttpUtils.getPalletLiftCommand(shuttleNo, wrkNo, true)); | 
|---|
|  |  |  | //        commands.addAll(result.getCommands());//起点到目标点移动命令 | 
|---|
|  |  |  | //        //当小车行走到目标点后,需要下发一条下降命令将货物放置 | 
|---|
|  |  |  | //        commands.add(NyHttpUtils.getPalletLiftCommand(shuttleNo, wrkNo, false)); | 
|---|
|  |  |  | //        return result(commands, nodes); | 
|---|
|  |  |  | //    } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //    /** | 
|---|
|  |  |  | //     * 计算并获取小车从起点——中间点——目标点之间搬运货物动作命令 | 
|---|
|  |  |  | //     * @param shuttleNo 小车号 | 
|---|
|  |  |  | //     * @param wrkNo 工作号 | 
|---|
|  |  |  | //     * @param startPoint 起点(小车当前位置) | 
|---|
|  |  |  | //     * @param middlePoint 中间点(货物位置) | 
|---|
|  |  |  | //     * @param targetPoint 目标点(货物目标位置) | 
|---|
|  |  |  | //     */ | 
|---|
|  |  |  | //    public static synchronized ShuttleOperaResult getShuttleTransportCommands(Integer shuttleNo, Integer wrkNo, String startPoint, String middlePoint, String targetPoint) { | 
|---|
|  |  |  | //        //行走路径 | 
|---|
|  |  |  | //        ArrayList<NavigateNode> nodes = new ArrayList<>(); | 
|---|
|  |  |  | //        //命令集合 | 
|---|
|  |  |  | //        ArrayList<NyShuttleHttpCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        if (!startPoint.equals(middlePoint)) {//起点和中间点不一致,需要计算起点到中间点路径 | 
|---|
|  |  |  | //            //计算起点到中间点命令 | 
|---|
|  |  |  | //            ShuttleOperaResult result1 = getStartToTargetCommands(shuttleNo, wrkNo, startPoint, middlePoint, NavigationMapType.NORMAL.id); | 
|---|
|  |  |  | //            if (result1 == null) { | 
|---|
|  |  |  | //                //计算结果必须不为空,否则计算失败 | 
|---|
|  |  |  | //                return null; | 
|---|
|  |  |  | //            } | 
|---|
|  |  |  | //            nodes.addAll(result1.getNodes()); | 
|---|
|  |  |  | //            commands.addAll(result1.getCommands()); | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        //计算中间点到目标点命令 | 
|---|
|  |  |  | //        ShuttleOperaResult result2 = getStartToTargetCommands(shuttleNo, wrkNo, middlePoint, targetPoint, NavigationMapType.DFX.id); | 
|---|
|  |  |  | //        if (result2 == null) { | 
|---|
|  |  |  | //            //计算结果必须不为空,否则计算失败 | 
|---|
|  |  |  | //            return null; | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | //        nodes.addAll(result2.getNodes()); | 
|---|
|  |  |  | //        //当小车行走到中间点后,需要下发一条顶升命令将货物进行搬运 | 
|---|
|  |  |  | //        commands.add(NyHttpUtils.getPalletLiftCommand(shuttleNo, wrkNo, true)); | 
|---|
|  |  |  | //        commands.addAll(result2.getCommands());//中间点到目标点移动命令 | 
|---|
|  |  |  | //        //当小车行走到目标点后,需要下发一条下降命令将货物放置 | 
|---|
|  |  |  | //        commands.add(NyHttpUtils.getPalletLiftCommand(shuttleNo, wrkNo, false)); | 
|---|
|  |  |  | //        return result(commands, nodes); | 
|---|
|  |  |  | //    } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //    /** | 
|---|
|  |  |  | //     * 获取起点到目标点行走命令 | 
|---|
|  |  |  | //     */ | 
|---|
|  |  |  | //    public synchronized ShuttleOperaResult getStartToTargetCommands(Integer shuttleNo, Integer wrkNo, String startPoint, String targetPoint, Integer mapType) { | 
|---|
|  |  |  | //        NavigateMapUtils navigateMapUtils = SpringUtils.getBean(NavigateMapUtils.class); | 
|---|
|  |  |  | //        //计算起点到目标点行走节点 | 
|---|
|  |  |  | //        List<NavigateNode> calc = NavigateUtils.calc(startPoint, targetPoint, mapType, Utils.getShuttlePoints(shuttleNo, Utils.getLev(startPoint)), null); | 
|---|
|  |  |  | //        if (calc == null) { | 
|---|
|  |  |  | //            return null; | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        //命令集合 | 
|---|
|  |  |  | //        ArrayList<NyShuttleHttpCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  | //        List<NavigateNode> allNode = new ArrayList<>(); | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        //获取分段路径 | 
|---|
|  |  |  | //        ArrayList<ArrayList<NavigateNode>> data = NavigateUtils.getSectionPath(calc); | 
|---|
|  |  |  | //        //将每一段路径分成command指令 | 
|---|
|  |  |  | //        for (ArrayList<NavigateNode> nodes : data) { | 
|---|
|  |  |  | //            //开始路径 | 
|---|
|  |  |  | //            NavigateNode startPath = nodes.get(0); | 
|---|
|  |  |  | //            //目标路径 | 
|---|
|  |  |  | //            NavigateNode targetPath = nodes.get(nodes.size() - 1); | 
|---|
|  |  |  | //            //获取移动命令 | 
|---|
|  |  |  | //            NyShuttleHttpCommand moveCommand = NyHttpUtils.getMoveCommand(shuttleNo, wrkNo, startPath, targetPath); | 
|---|
|  |  |  | //            moveCommand.setNodes(nodes);//将行走节点添加到每一步命令中 | 
|---|
|  |  |  | //            commands.add(moveCommand); | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //            allNode.addAll(nodes); | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        boolean result = navigateMapUtils.writeNavigateNodeToRedisMap(Utils.getLev(startPoint), shuttleNo, allNode, true);//锁定路径 | 
|---|
|  |  |  | //        if (!result) { | 
|---|
|  |  |  | //            return null;//路径锁定失败 | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | //        return result(commands, calc); | 
|---|
|  |  |  | //    } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //    /** | 
|---|
|  |  |  | //     * 获取起点到目标点行走命令(可传白名单) | 
|---|
|  |  |  | //     */ | 
|---|
|  |  |  | //    public static synchronized ShuttleOperaResult getStartToTargetCommandsByWhites(Integer shuttleNo, Integer wrkNo, String startPoint, String targetPoint, Integer mapType, List<int[]> whites) { | 
|---|
|  |  |  | //        NavigateMapUtils navigateMapUtils = SpringUtils.getBean(NavigateMapUtils.class); | 
|---|
|  |  |  | //        //计算起点到目标点行走节点 | 
|---|
|  |  |  | //        List<NavigateNode> calc = NavigateUtils.calc(startPoint, targetPoint, mapType, Utils.getShuttlePoints(shuttleNo, Utils.getLev(startPoint)), whites); | 
|---|
|  |  |  | //        if (calc == null) { | 
|---|
|  |  |  | //            return null; | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        //命令集合 | 
|---|
|  |  |  | //        ArrayList<NyShuttleHttpCommand> commands = new ArrayList<>(); | 
|---|
|  |  |  | //        List<NavigateNode> allNode = new ArrayList<>(); | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        //获取分段路径 | 
|---|
|  |  |  | //        ArrayList<ArrayList<NavigateNode>> data = NavigateUtils.getSectionPath(calc); | 
|---|
|  |  |  | //        //将每一段路径分成command指令 | 
|---|
|  |  |  | //        for (ArrayList<NavigateNode> nodes : data) { | 
|---|
|  |  |  | //            //开始路径 | 
|---|
|  |  |  | //            NavigateNode startPath = nodes.get(0); | 
|---|
|  |  |  | //            //目标路径 | 
|---|
|  |  |  | //            NavigateNode targetPath = nodes.get(nodes.size() - 1); | 
|---|
|  |  |  | //            //获取移动命令 | 
|---|
|  |  |  | //            NyShuttleHttpCommand moveCommand = NyHttpUtils.getMoveCommand(shuttleNo, wrkNo, startPath, targetPath); | 
|---|
|  |  |  | //            moveCommand.setNodes(nodes);//将行走节点添加到每一步命令中 | 
|---|
|  |  |  | //            commands.add(moveCommand); | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //            allNode.addAll(nodes); | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //        //锁定路径时剔除白名单节点 | 
|---|
|  |  |  | //        ArrayList<NavigateNode> nodes = new ArrayList<>(); | 
|---|
|  |  |  | //        for (NavigateNode node : allNode) { | 
|---|
|  |  |  | //            boolean flag = false; | 
|---|
|  |  |  | //            for (int[] white : whites) { | 
|---|
|  |  |  | //                if (node.getX() == white[0] && node.getY() == white[1]) { | 
|---|
|  |  |  | //                    flag = true;//存在白名单节点 | 
|---|
|  |  |  | //                    break;//跳过白名单节点 | 
|---|
|  |  |  | //                } | 
|---|
|  |  |  | //            } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //            if (!flag) { | 
|---|
|  |  |  | //                nodes.add(node); | 
|---|
|  |  |  | //            } | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | //        boolean result = navigateMapUtils.writeNavigateNodeToRedisMap(Utils.getLev(startPoint), shuttleNo, nodes, true);//锁定路径 | 
|---|
|  |  |  | //        if (!result) { | 
|---|
|  |  |  | //            return null;//路径锁定失败 | 
|---|
|  |  |  | //        } | 
|---|
|  |  |  | //        return result(commands, calc); | 
|---|
|  |  |  | //    } | 
|---|
|  |  |  | // | 
|---|
|  |  |  | //    //返回结果集 | 
|---|
|  |  |  | //    public static ShuttleOperaResult result(List<NyShuttleHttpCommand> commands, List<NavigateNode> nodes) { | 
|---|
|  |  |  | //        return new ShuttleOperaResult(commands, nodes); | 
|---|
|  |  |  | //    } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|