|  |  |  | 
|---|
|  |  |  | package com.zy.common.utils; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import com.alibaba.fastjson.JSON; | 
|---|
|  |  |  | import com.baomidou.mybatisplus.mapper.EntityWrapper; | 
|---|
|  |  |  | import com.core.common.SpringUtils; | 
|---|
|  |  |  | import com.core.exception.CoolException; | 
|---|
|  |  |  | import com.zy.asrs.entity.BasDevp; | 
|---|
|  |  |  | import com.zy.asrs.entity.BasShuttle; | 
|---|
|  |  |  | import com.zy.asrs.entity.WrkCharge; | 
|---|
|  |  |  | import com.zy.asrs.entity.WrkMast; | 
|---|
|  |  |  | import com.zy.asrs.mapper.WrkChargeMapper; | 
|---|
|  |  |  | import com.zy.asrs.mapper.WrkMastMapper; | 
|---|
|  |  |  | import com.zy.asrs.service.BasDevpService; | 
|---|
|  |  |  | import com.zy.asrs.service.BasShuttleService; | 
|---|
|  |  |  | import com.zy.asrs.service.WrkMastService; | 
|---|
|  |  |  | import com.zy.asrs.utils.Utils; | 
|---|
|  |  |  | import com.zy.common.model.NavigateNode; | 
|---|
|  |  |  | 
|---|
|  |  |  | import com.zy.common.service.CommonService; | 
|---|
|  |  |  | import com.zy.core.News; | 
|---|
|  |  |  | import com.zy.core.cache.SlaveConnection; | 
|---|
|  |  |  | import com.zy.core.enums.ShuttleChargeType; | 
|---|
|  |  |  | import com.zy.core.enums.SlaveType; | 
|---|
|  |  |  | import com.zy.core.model.LiftSlave; | 
|---|
|  |  |  | import com.zy.core.model.ShuttleSlave; | 
|---|
|  |  |  | import com.zy.core.model.protocol.LiftProtocol; | 
|---|
|  |  |  | import com.zy.core.model.protocol.LiftStaProtocol; | 
|---|
|  |  |  | import com.zy.core.model.protocol.NyShuttleProtocol; | 
|---|
|  |  |  | import com.zy.core.properties.SlaveProperties; | 
|---|
|  |  |  | import com.zy.core.thread.LiftThread; | 
|---|
|  |  |  | import com.zy.core.thread.NyShuttleThread; | 
|---|
|  |  |  | import com.zy.system.entity.Config; | 
|---|
|  |  |  | import com.zy.system.service.ConfigService; | 
|---|
|  |  |  | import org.springframework.beans.factory.annotation.Autowired; | 
|---|
|  |  |  | import org.springframework.stereotype.Service; | 
|---|
|  |  |  | import org.springframework.transaction.annotation.Transactional; | 
|---|
|  |  |  | 
|---|
|  |  |  | * 调度车辆 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public boolean dispatchShuttle(Integer wrkNo, String locNo) { | 
|---|
|  |  |  | //检测目标库位组是否存在小车,如存在小车则直接指定该车 | 
|---|
|  |  |  | WrkMast wrkMast = wrkMastMapper.selectByWorkNo(wrkNo); | 
|---|
|  |  |  | if (wrkMast != null) { | 
|---|
|  |  |  | String targetLocNo = wrkMast.getIoType() < 100 ? wrkMast.getLocNo() : wrkMast.getSourceLocNo(); | 
|---|
|  |  |  | List<String> groupLoc = Utils.getGroupLoc(targetLocNo); | 
|---|
|  |  |  | Integer groupShuttleNo = Utils.checkGroupLocHasShuttle(groupLoc); | 
|---|
|  |  |  | if (groupShuttleNo != null) { | 
|---|
|  |  |  | //存在小车,直接调度该车 | 
|---|
|  |  |  | return shuttleMoveGenerate(wrkNo, locNo, groupShuttleNo); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | ArrayList<NyShuttleThread> sameLev = new ArrayList<>();//相同楼层的穿梭车 | 
|---|
|  |  |  | ArrayList<NyShuttleThread> diffLev = new ArrayList<>();//不同楼层的穿梭车 | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (currentLocNo.equals(locNo)) { | 
|---|
|  |  |  | //车辆当前位置已经是目标库位,调度该车 | 
|---|
|  |  |  | shuttleMoveGenerate(wrkNo, locNo, shuttleProtocol.getShuttleNo().intValue()); | 
|---|
|  |  |  | //给工作档绑定小车号 | 
|---|
|  |  |  | WrkMast wrkMast1 = wrkMastMapper.selectByWorkNo(wrkNo); | 
|---|
|  |  |  | if (wrkMast1 != null) { | 
|---|
|  |  |  | wrkMast1.setShuttleNo(shuttleProtocol.getShuttleNo().intValue()); | 
|---|
|  |  |  | wrkMastMapper.updateById(wrkMast1); | 
|---|
|  |  |  | return true; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | //当前穿梭车库位号 | 
|---|
|  |  |  | String currentLocNo = shuttleThread.getShuttleProtocol().getCurrentLocNo(); | 
|---|
|  |  |  | //当前穿梭车线程到目标地点距离 | 
|---|
|  |  |  | List<NavigateNode> currentShuttlePath = NavigateUtils.calc(currentLocNo, locNo, NavigationMapType.NORMAL.id, Utils.getShuttlePoints(shuttleThread.getSlave().getId(), Utils.getLev(currentLocNo)));//搜索空闲穿梭车,使用正常通道地图 | 
|---|
|  |  |  | List<NavigateNode> currentShuttlePath = NavigateUtils.calc(currentLocNo, locNo, NavigationMapType.NORMAL.id, Utils.getShuttlePoints(shuttleThread.getSlave().getId(), Utils.getLev(currentLocNo)), null);//搜索空闲穿梭车,使用正常通道地图 | 
|---|
|  |  |  | if (currentShuttlePath == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | if (currentAllDistance < recentAllDistance) { | 
|---|
|  |  |  | //如果当前楼层的车路径更小,则更新最近穿梭车 | 
|---|
|  |  |  | recentShuttle = shuttleThread; | 
|---|
|  |  |  | recentAllDistance = currentAllDistance; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | //同一楼层,没有空闲穿梭车,只能从其他楼层调度 | 
|---|
|  |  |  | //寻找离任务最近的穿梭车 | 
|---|
|  |  |  | for (NyShuttleThread shuttleThread : diffLev) { | 
|---|
|  |  |  | //寻找离任务楼层最近的穿梭车(不考虑跨楼层小车移动距离) | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //当前穿梭车库位号 | 
|---|
|  |  |  | String currentLocNo = shuttleThread.getShuttleProtocol().getCurrentLocNo(); | 
|---|
|  |  |  | int currentLev = Utils.getLev(currentLocNo); | 
|---|
|  |  |  | List<WrkMast> wrkMasts1 = wrkMastService.selectNoShuttleWrkByLev(currentLev);//判断当前穿梭车楼层是否有待分配车辆的任务,如果有则不分配这辆车 | 
|---|
|  |  |  | if (wrkMasts1.size() > 0) { | 
|---|
|  |  |  | //存在其他任务,跳过这辆车 | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | //获取任务 | 
|---|
|  |  |  | WrkMast wrkMast1 = wrkMastMapper.selectByWorkNo(wrkNo); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (wrkMast1 != null) { | 
|---|
|  |  |  | String targetLoc = wrkMast1.getIoType() < 100 ? wrkMast1.getLocNo() : wrkMast1.getSourceLocNo(); | 
|---|
|  |  |  | int lev = Utils.getLev(targetLoc);//目标楼层 | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //检测目标楼层车数量是否小于允许的最大数量 | 
|---|
|  |  |  | boolean checkDispatchMaxNum = checkDispatchMaxNum(lev); | 
|---|
|  |  |  | if (!checkDispatchMaxNum) { | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //当前穿梭车线程到当前车子所在楼层的提升机口距离 | 
|---|
|  |  |  | List<NavigateNode> currentShuttlePath = NavigateUtils.calc(currentLocNo, Utils.levToOutInStaLocNo(currentLev), NavigationMapType.NORMAL.id, Utils.getShuttlePoints(shuttleThread.getSlave().getId(), currentLev));//搜索空闲穿梭车,使用正常通道地图 | 
|---|
|  |  |  | if (currentShuttlePath == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | int recentValue = 99999;//最小差值 | 
|---|
|  |  |  | for (NyShuttleThread shuttleThread : diffLev) { | 
|---|
|  |  |  | //当前穿梭车库位号 | 
|---|
|  |  |  | String currentLocNo = shuttleThread.getShuttleProtocol().getCurrentLocNo(); | 
|---|
|  |  |  | int currentLev = Utils.getLev(currentLocNo); | 
|---|
|  |  |  | List<WrkMast> wrkMasts1 = wrkMastService.selectNoShuttleWrkByLev(currentLev);//判断当前穿梭车楼层是否有待分配车辆的任务,如果有则不分配这辆车 | 
|---|
|  |  |  | if (wrkMasts1.size() > 0) { | 
|---|
|  |  |  | //存在其他任务,跳过这辆车 | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Integer currentAllDistance = NavigateUtils.getOriginPathAllDistance(currentShuttlePath);//计算当前路径行走总距离 | 
|---|
|  |  |  | if (currentAllDistance < recentAllDistance) { | 
|---|
|  |  |  | //如果当前楼层的车路径更小,则更新最近穿梭车 | 
|---|
|  |  |  | recentShuttle = shuttleThread; | 
|---|
|  |  |  | //ABS(目标楼层 - 当前楼层) 得到差距,取最小差值 | 
|---|
|  |  |  | int currentValue = Math.abs(lev - currentLev); | 
|---|
|  |  |  | if (currentValue < recentValue) { | 
|---|
|  |  |  | //如果当前楼层的车路径更小,则更新最近穿梭车 | 
|---|
|  |  |  | recentShuttle = shuttleThread; | 
|---|
|  |  |  | recentValue = currentValue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | //判断是否有充电任务正在使用穿梭车 | 
|---|
|  |  |  | WrkCharge wrkCharge = wrkChargeMapper.selectWorking(shuttleNo); | 
|---|
|  |  |  | if (wrkCharge != null) {//小车存在充电任务,等待执行完成后再生成新的任务 | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | //判断目标点是否为充电桩,如果是去充电则放行 | 
|---|
|  |  |  | boolean toCharge = false;//去充电目标 | 
|---|
|  |  |  | for (ShuttleChargeType chargeType : ShuttleChargeType.values()) { | 
|---|
|  |  |  | if (chargeType.locNo.equals(locNo)) { | 
|---|
|  |  |  | toCharge = true;//去充电桩 | 
|---|
|  |  |  | break; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (wrkCharge.getWrkSts() == 53) { | 
|---|
|  |  |  | toCharge = true;//充电结束,允许生成移库任务 | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!toCharge) { | 
|---|
|  |  |  | //不是去充电桩且存在充电任务,禁止生成新的移动任务 | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Integer staNo = null; | 
|---|
|  |  |  | if (Utils.getLev(locNo) == shuttleProtocol.getPoint().getZ()) { | 
|---|
|  |  |  | Integer sourceStaNo = null;//小车换层源站点 | 
|---|
|  |  |  | Integer staNo = null;//小车换层目标站点 | 
|---|
|  |  |  | if (Utils.getLev(locNo) != shuttleProtocol.getPoint().getZ()) { | 
|---|
|  |  |  | //目标库位和小车库位处于不同一楼层,需要通过提升机调度 | 
|---|
|  |  |  | //获取穿梭车最近且空闲的提升机输送站点 | 
|---|
|  |  |  | BasDevp liftSta = this.getRecentLiftSta(shuttleNo); | 
|---|
|  |  |  | LiftStaProtocol liftSta = this.getRecentLiftSta(shuttleNo, Utils.getLev(locNo)); | 
|---|
|  |  |  | if (liftSta == null) { | 
|---|
|  |  |  | return false;//没有可用且空闲的输送站点 | 
|---|
|  |  |  | } | 
|---|
|  |  |  | staNo = liftSta.getDevNo(); | 
|---|
|  |  |  | sourceStaNo = liftSta.getStaNo();//源站点 | 
|---|
|  |  |  | //提升机号*100+目标楼层=目标站点 | 
|---|
|  |  |  | staNo = liftSta.getLiftNo() * 100 + Utils.getLev(locNo);//目标站 | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 获取工作号 | 
|---|
|  |  |  | 
|---|
|  |  |  | wrkMast.setShuttleNo(shuttleNo);//穿梭车号 | 
|---|
|  |  |  | wrkMast.setSourceLocNo(shuttleProtocol.getCurrentLocNo()); // 源库位 => 小车当前库位号 | 
|---|
|  |  |  | wrkMast.setLocNo(locNo); // 目标库位 | 
|---|
|  |  |  | wrkMast.setSourceStaNo(sourceStaNo);//源站 | 
|---|
|  |  |  | wrkMast.setStaNo(staNo);//目标站 | 
|---|
|  |  |  | wrkMast.setPicking("N"); // 拣料 | 
|---|
|  |  |  | wrkMast.setExitMk("N"); // 退出 | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 检测目标楼层车数量是否小于允许的最大数量 | 
|---|
|  |  |  | * true: 小于最大数量  false: 大于或等于最大数量 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public boolean checkDispatchMaxNum(Integer lev) { | 
|---|
|  |  |  | BasShuttleService basShuttleService = SpringUtils.getBean(BasShuttleService.class); | 
|---|
|  |  |  | ConfigService configService = SpringUtils.getBean(ConfigService.class); | 
|---|
|  |  |  | EntityWrapper<Config> wrapper = new EntityWrapper<>(); | 
|---|
|  |  |  | wrapper.eq("code", "dispatchShuttleMaxNum"); | 
|---|
|  |  |  | Config config = configService.selectOne(wrapper); | 
|---|
|  |  |  | if (config == null) { | 
|---|
|  |  |  | return false; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int levCount = 0;//目标楼层车辆数量 | 
|---|
|  |  |  | for (ShuttleSlave shuttle : slaveProperties.getShuttle()) { | 
|---|
|  |  |  | //获取四向穿梭车线程 | 
|---|
|  |  |  | NyShuttleThread shuttleThread = (NyShuttleThread) SlaveConnection.get(SlaveType.Shuttle, shuttle.getId()); | 
|---|
|  |  |  | NyShuttleProtocol shuttleProtocol = shuttleThread.getShuttleProtocol(); | 
|---|
|  |  |  | if (shuttleProtocol == null || shuttleProtocol.getShuttleNo() == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | NyShuttleProtocol.NyShuttlePointClass point = null; | 
|---|
|  |  |  | if (shuttleProtocol.getPoint() == null) { | 
|---|
|  |  |  | BasShuttle basShuttle = basShuttleService.selectById(shuttle.getId());//小车如果没有数据,从数据库取数据 | 
|---|
|  |  |  | if (basShuttle == null || basShuttle.getPoint() == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | point = JSON.parseObject(basShuttle.getPoint(), NyShuttleProtocol.NyShuttlePointClass.class); | 
|---|
|  |  |  | }else { | 
|---|
|  |  |  | point = shuttleProtocol.getPoint(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (point.getZ().equals(lev)) { | 
|---|
|  |  |  | levCount++;//目标楼层有车,数量增加 | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //搜索是否存在前往目标楼层的小车移动工作档 | 
|---|
|  |  |  | for (WrkMast wrkMast : wrkMastMapper.selectShuttleMoveWrk()) { | 
|---|
|  |  |  | if (wrkMast.getSourceLocNo() == null || wrkMast.getLocNo() == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int sourceLev = Utils.getLev(wrkMast.getSourceLocNo());//工作档源楼层 | 
|---|
|  |  |  | int targetLev = Utils.getLev(wrkMast.getLocNo());//工作档目标楼层 | 
|---|
|  |  |  | if (sourceLev == lev) { | 
|---|
|  |  |  | continue;//工作档楼层和目标楼层相同,跳过 | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (targetLev == lev) { | 
|---|
|  |  |  | levCount++;//工作档目标楼层和实际楼层相同,数量增加 | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return levCount < Integer.parseInt(config.getValue()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 获取穿梭车最近且空闲的提升机输送站点 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public BasDevp getRecentLiftSta(Integer shuttleNo) { | 
|---|
|  |  |  | public LiftStaProtocol getRecentLiftSta(Integer shuttleNo, Integer targetLev) { | 
|---|
|  |  |  | //获取四向穿梭车线程 | 
|---|
|  |  |  | NyShuttleThread shuttleThread = (NyShuttleThread) SlaveConnection.get(SlaveType.Shuttle, shuttleNo); | 
|---|
|  |  |  | if (shuttleThread == null) { | 
|---|
|  |  |  | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //获取小车同一楼层的站点 | 
|---|
|  |  |  | ArrayList<LiftStaProtocol> list = new ArrayList<>(); | 
|---|
|  |  |  | int lev = Utils.getLev(shuttleProtocol.getCurrentLocNo());//小车楼层 | 
|---|
|  |  |  | for (LiftSlave slave : slaveProperties.getLift()) { | 
|---|
|  |  |  | LiftThread liftThread = (LiftThread) SlaveConnection.get(SlaveType.Lift, slave.getId()); | 
|---|
|  |  |  | if (liftThread == null) { | 
|---|
|  |  |  | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | BasDevp basDevp = basDevpService.selectByLevAndLiftNo(Utils.getLev(shuttleProtocol.getCurrentLocNo()), slave.getId()); | 
|---|
|  |  |  | if (basDevp == null) { | 
|---|
|  |  |  | LiftStaProtocol liftStaProtocol = NyLiftUtils.getLiftStaByLev(slave.getId(), lev); | 
|---|
|  |  |  | if (liftStaProtocol == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return basDevp; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //判断目标楼层站点是否无托盘 | 
|---|
|  |  |  | LiftStaProtocol targetLiftStaProtocol = NyLiftUtils.getLiftStaByLev(slave.getId(), targetLev); | 
|---|
|  |  |  | if (targetLiftStaProtocol == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (targetLiftStaProtocol.getHasTray()) { | 
|---|
|  |  |  | continue;//有托盘跳过 | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | list.add(liftStaProtocol); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | if (list.isEmpty()) { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | String currentLocNo = shuttleProtocol.getCurrentLocNo();//小车位置 | 
|---|
|  |  |  | Integer recentAllDistance = 9999999; | 
|---|
|  |  |  | LiftStaProtocol recentSta = null;//最近站点 | 
|---|
|  |  |  | //搜索距离小车最近的站点 | 
|---|
|  |  |  | for (LiftStaProtocol liftStaProtocol : list) { | 
|---|
|  |  |  | Integer staNo = liftStaProtocol.getStaNo();//站点号 | 
|---|
|  |  |  | String locNo = liftStaProtocol.getLocNo();//站点库位号 | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //当前穿梭车线程到目标地点距离 | 
|---|
|  |  |  | List<NavigateNode> currentShuttlePath = NavigateUtils.calc(currentLocNo, locNo, NavigationMapType.NORMAL.id, Utils.getShuttlePoints(shuttleNo, Utils.getLev(currentLocNo)), null);//使用正常通道地图 | 
|---|
|  |  |  | if (currentShuttlePath == null) { | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | Integer currentAllDistance = NavigateUtils.getOriginPathAllDistance(currentShuttlePath);//计算当前路径行走总距离 | 
|---|
|  |  |  | if (currentAllDistance < recentAllDistance) { | 
|---|
|  |  |  | //如果当前楼层的车路径更小,则更新最近站点 | 
|---|
|  |  |  | recentSta = liftStaProtocol; | 
|---|
|  |  |  | recentAllDistance = currentAllDistance; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return recentSta; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|