package com.pz.system.service.impl;

import cn.hutool.core.bean.BeanUtil;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.core.toolkit.StringUtils;
import com.pz.common.core.page.TableDataInfo;
import com.pz.common.core.domain.PageQuery;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import com.pz.common.exception.ServiceException;
import com.pz.common.utils.redis.RedisUtils;
import com.pz.merchant.domain.Employees;
import com.pz.merchant.domain.vo.SonOrderVo;
import com.pz.merchant.mapper.EmployeesMapper;
import com.pz.merchant.service.ISonOrderService;
import com.pz.system.datastructure.OrderDelayQueue;
import com.pz.system.datastructure.TotalOrderDelayOperator;
import com.pz.system.domain.DbwzOrder;
import com.pz.system.domain.TotalOrder;
import com.pz.system.domain.YypzOrder;
import com.pz.system.domain.bo.OrderFinishedBo;
import com.pz.system.mapper.TotalOrderMapper;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import com.pz.system.domain.bo.ZyphOrderBo;
import com.pz.system.domain.vo.ZyphOrderVo;
import com.pz.system.domain.ZyphOrder;
import com.pz.system.mapper.ZyphOrderMapper;
import com.pz.system.service.IZyphOrderService;
import org.springframework.transaction.annotation.Transactional;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.temporal.ChronoUnit;
import java.util.*;
import java.util.concurrent.TimeUnit;

/**
 * 住院陪诊订单Service业务层处理
 *
 * @author ruoyi
 * @date 2023-09-11
 */
@RequiredArgsConstructor
@Service
@Slf4j
public class ZyphOrderServiceImpl implements IZyphOrderService, ISonOrderService {

    private final ZyphOrderMapper baseMapper;
    private final TotalOrderMapper totalOrderMapper;
    private final EmployeesMapper employeesMapper;
    private final OrderDelayQueue delayQueue;

    private static final SimpleDateFormat DATETIME_FORMAT = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    /**
     * 查询住院陪诊订单
     */
    @Override
    public ZyphOrderVo queryById(Integer id) {
        return baseMapper.selectVoById(id);
    }

    /**
     * 查询住院陪诊订单列表
     */
    @Override
    public TableDataInfo<ZyphOrderVo> queryPageList(ZyphOrderBo bo, PageQuery pageQuery) {
        Page<ZyphOrderVo> result = baseMapper.findZyphOrderVoPage(pageQuery.build(), bo);
        return TableDataInfo.build(result);
    }

    /**
     * 查询住院陪诊订单列表
     */
    @Override
    public List<ZyphOrderVo> queryList(ZyphOrderBo bo) {
        LambdaQueryWrapper<ZyphOrder> lqw = buildQueryWrapper(bo);
        return baseMapper.selectVoList(lqw);
    }

    private LambdaQueryWrapper<ZyphOrder> buildQueryWrapper(ZyphOrderBo bo) {
        Map<String, Object> params = bo.getParams();
        LambdaQueryWrapper<ZyphOrder> lqw = Wrappers.lambdaQuery();
        lqw.eq(bo.getOrderId() != null, ZyphOrder::getOrderId, bo.getOrderId());
        lqw.eq(bo.getHid() != null, ZyphOrder::getHid, bo.getHid());
        lqw.eq(bo.getDid() != null, ZyphOrder::getDid, bo.getDid());
        lqw.eq(bo.getVisitor() != null, ZyphOrder::getVisitor, bo.getVisitor());
        lqw.eq(StringUtils.isNotBlank(bo.getStartDay()), ZyphOrder::getStartDay, bo.getStartDay());
        lqw.eq(StringUtils.isNotBlank(bo.getEndDay()), ZyphOrder::getEndDay, bo.getEndDay());
        lqw.eq(StringUtils.isNotBlank(bo.getStartTime()), ZyphOrder::getStartTime, bo.getStartTime());
        lqw.eq(StringUtils.isNotBlank(bo.getEndTime()), ZyphOrder::getEndTime, bo.getEndTime());
        lqw.eq(bo.getStatus() != null, ZyphOrder::getStatus, bo.getStatus());
        lqw.eq(bo.getOverTime() != null, ZyphOrder::getOverTime, bo.getOverTime());
        lqw.eq(StringUtils.isNotBlank(bo.getVoucher()), ZyphOrder::getVoucher, bo.getVoucher());
        return lqw;
    }

    /**
     * 新增住院陪诊订单
     */
    @Override
    public Boolean insertByBo(ZyphOrderBo bo) {
        ZyphOrder add = BeanUtil.toBean(bo, ZyphOrder.class);
        validEntityBeforeSave(add);
        boolean flag = baseMapper.insert(add) > 0;
        if (flag) {
            bo.setId(add.getId());
        }
        return flag;
    }

    /**
     * 修改住院陪诊订单
     */
    @Override
    public Boolean updateByBo(ZyphOrderBo bo) {
        ZyphOrder update = BeanUtil.toBean(bo, ZyphOrder.class);
        validEntityBeforeSave(update);
        return baseMapper.updateById(update) > 0;
    }

    /**
     * 保存前的数据校验
     */
    private void validEntityBeforeSave(ZyphOrder entity) {
        // TODO 做一些数据校验,如唯一约束
    }

    /**
     * 批量删除住院陪诊订单
     */
    @Override
    public Boolean deleteWithValidByIds(Collection<Integer> ids, Boolean isValid) {
        if (isValid) {
            // TODO 做一些业务上的校验,判断是否需要校验
        }
        return baseMapper.deleteBatchIds(ids) > 0;
    }

    /**
     * 根据主订单ID查询子订单相关信息
     *
     * @param totalId 主订单ID
     * @return 子订单相关信息
     */
    @Override
    public SonOrderVo getSonOrderSimpleDataByTotalId(Integer totalId) {
        return baseMapper.selectSonOrderInfoByTotalId(totalId);
    }

    /**
     * 切换子订单状态
     *
     * @param totalId 主订单ID
     * @param target  子订单目标状态
     * @return 操作结果
     */
    @Override
    @Transactional(rollbackFor = Exception.class)
    public boolean switchSonOrderStatus(Long totalId, Integer target) {
        ZyphOrder sonOrder = baseMapper.selectOne(Wrappers.<ZyphOrder>lambdaQuery().eq(ZyphOrder::getOrderId, totalId));
        Objects.requireNonNull(sonOrder, "子订单不存在,请检查");
        sonOrder.setStatus(target);
        totalOrderMapper.update(null,new LambdaUpdateWrapper<TotalOrder>().set(TotalOrder::getSuborderStatus,target).eq(TotalOrder::getId,totalId));
        return baseMapper.updateById(sonOrder) > 0;
    }

    /**
     * 查询住院陪护订单详情
     *
     * @param totalId 主订单ID
     * @return 订单详情
     */
    @Override
    public Object getSonOrderDetailDataByTotalId(Integer totalId) {
        return baseMapper.selectZyphOrderDetailDataByTotalId(totalId);
    }


    /**
     * 陪诊员取消住院陪护订单
     *
     * @param totalId 取消订单
     * @return 操作结果
     */
    @Override
    @Transactional(rollbackFor = Exception.class)
    public boolean accompanyCancellationOfOrder(Integer totalId) {
        TotalOrder totalOrder = totalOrderMapper.selectById(totalId);
        Objects.requireNonNull(totalOrder, "主订单不存在");
        Integer emId = totalOrder.getEmId();
        if (emId == 0) {
            throw new ServiceException("订单暂未分配陪诊员,拒绝该操作");
        }
        ZyphOrder zyphOrder = baseMapper.selectOne(Wrappers.<ZyphOrder>lambdaQuery().eq(ZyphOrder::getOrderId, totalId));
        Objects.requireNonNull(zyphOrder, "子订单与主订单不一致!");
        String dateTimeStr = zyphOrder.getStartDay() + " " + zyphOrder.getEndTime();
        Date paseDateTime = null;
        try {
            paseDateTime = DATETIME_FORMAT.parse(dateTimeStr);
        } catch (ParseException e) {
            log.error("时间解析失败");
        }
        // 取消次数
        Integer cancel = RedisUtils.<Integer>getCacheMapValue(ISonOrderService.ORDER_CANCEL_CACHE_PREFIX, String.valueOf(emId));
        if (paseDateTime != null) {
            LocalDateTime visitTime = LocalDateTime.ofInstant(paseDateTime.toInstant(), ZoneId.systemDefault());
            LocalDateTime currentDate = LocalDateTime.ofInstant(new Date().toInstant(), ZoneId.systemDefault());
            // 若缓存为空,则查询数据表
            if (cancel == null) {
                cancel = Optional.ofNullable(employeesMapper.selectById(emId).getKillOrder()).orElse(0);
            }
            // 若在18小时前取消订单,需记录取消次数
            if (visitTime.isBefore(currentDate) || ChronoUnit.HOURS.between(visitTime, currentDate) <= 18) {
                cancel++;
            }
        }
        // 更新主订单数据
        totalOrder.setEmId(0);
        totalOrder.setSuborderStatus(0);
        if (totalOrderMapper.updateById(totalOrder) < 0) {
            throw new ServiceException("更新主订单失败");
        }
        // 更新子订单
        zyphOrder.setStatus(0);
        if (baseMapper.updateById(zyphOrder) < 0) {
            throw new ServiceException("子订单更新失败");
        }
        // 若取消次数大于3,则冻结用户账号
        if (cancel != null && cancel > 3) {
            if (employeesMapper.update(null,
                Wrappers.<Employees>lambdaUpdate()
                    .set(Employees::getStatus, 2)
                    .set(Employees::getKillOrder, cancel)
                    .eq(Employees::getId, emId)) < 0) {
                throw new ServiceException("用户冻结失败");
            }
        }
        if (cancel != null) {
            RedisUtils.setCacheMapValue(ISonOrderService.ORDER_CANCEL_CACHE_PREFIX, String.valueOf(emId), cancel);
        }
        return true;
    }

    /**
     * 陪诊员开始服务住院陪护订单
     *
     * @param totalId 主订单ID
     * @return 操作结果
     */
    @Override
    @Transactional(rollbackFor = Exception.class)
    public boolean accompanyStartService(Integer totalId) {
        TotalOrder totalOrder = totalOrderMapper.selectById(totalId);
        Objects.requireNonNull(totalOrder, "主订单不存在");
        if (totalOrder.getStatus() != 1 || totalOrder.getEmId() == 0) { // 用户已付款
            throw new ServiceException("主订单不符合开始服务要求");
        }
        ZyphOrder suborder = baseMapper.selectOne(Wrappers.<ZyphOrder>lambdaQuery().eq(ZyphOrder::getOrderId, totalId));
        Objects.requireNonNull(suborder, "子订单不存在");
        if (suborder.getStatus() != 1) { // 订单已接单
            throw new ServiceException("子订单不符合开始服务要求");
        }
        String dateTimeStr = suborder.getStartDay() + " " + suborder.getStartTime();
        Date paseDateTime = null;
        try {
            paseDateTime = DATETIME_FORMAT.parse(dateTimeStr);
        } catch (ParseException e) {
            log.error("时间解析失败");
        }
        // 操作时间若未到达开始时间,则拒绝该操作
        if (paseDateTime != null) {
            LocalDateTime visitTime = LocalDateTime.ofInstant(paseDateTime.toInstant(), ZoneId.systemDefault());
            LocalDateTime currentDate = LocalDateTime.ofInstant(new Date().toInstant(), ZoneId.systemDefault());
            if (currentDate.isBefore(visitTime)) {
                throw new ServiceException("请在到达陪诊时间后再点击开始服务");
            }
        }
        // 修改子订单状态为开始服务
        suborder.setStatus(6);
        totalOrder.setSuborderStatus(6);
        if (baseMapper.updateById(suborder) < 0) {
            throw new ServiceException("操作失败,子订单异常");
        }
        if (totalOrderMapper.updateById(totalOrder) < 0) {
            throw new ServiceException("操作失败,主订单异常");
        }
        return true;
    }

    /**
     * 陪诊员完成服务 住院陪护
     *
     * @param orderData 主订单ID
     * @return 操作结果
     */
    @Override
    @Transactional(rollbackFor = Exception.class)
    public boolean finishedService(OrderFinishedBo orderData) {
        TotalOrder totalOrder = totalOrderMapper.selectById(orderData.getOrderId());
        Objects.requireNonNull(totalOrder, "主订单不存在");
        if (totalOrder.getStatus() != 1 || totalOrder.getEmId() == 0) { // 用户已付款
            throw new ServiceException("主订单不符合开始服务要求");
        }
        ZyphOrder suborder = baseMapper.selectOne(Wrappers.<ZyphOrder>lambdaQuery().eq(ZyphOrder::getOrderId, totalOrder.getId()));
        Objects.requireNonNull(suborder, "子订单不存在");
        if (suborder.getStatus() != 6) { // 订单已接单
            throw new ServiceException("子订单不符合开始服务要求");
        }
        // 修改子订单状态为完成服务
        suborder.setStatus(2);
        suborder.setOverTime(new Date());
        suborder.setVoucher(orderData.getVoucher());
        suborder.setRemark(orderData.getRemark());
        totalOrder.setSuborderStatus(2);
        if (baseMapper.updateById(suborder) < 0) {
            throw new ServiceException("操作失败,子订单异常");
        }
        if (totalOrderMapper.updateById(totalOrder) < 0) {
            throw new ServiceException("操作失败,主订单异常");
        }
        // 子订单完成后,24小时后自动将主订单状态修改为完成
        long expirationTime = System.currentTimeMillis() + TimeUnit.MINUTES.toMillis(24);
        TotalOrderDelayOperator task = new TotalOrderDelayOperator(totalOrder.getId(), expirationTime, id -> {
            totalOrderMapper.update(null,
                Wrappers.<TotalOrder>lambdaUpdate().set(TotalOrder::getStatus, 2).eq(TotalOrder::getId, id));
        });
        delayQueue.addOrder(task);
        return true;
    }

    @Override
    public String getFinishTime(Integer totalId) {
        return baseMapper.selectFinishTime(totalId);
    }
}