#region << 版 本 注 释 >>
/*----------------------------------------------------------------
 * 命名空间:WIDESEAWCS_TaskInfoService
 * 创建者:胡童庆
 * 创建时间:2024/8/2 16:13:36
 * 版本:V1.0.0
 * 描述:
 *
 * ----------------------------------------------------------------
 * 修改人:
 * 修改时间:
 * 版本:V1.0.1
 * 修改说明:
 * 
 *----------------------------------------------------------------*/
#endregion << 版 本 注 释 >>

using System.Collections.Generic;
using System.Text;
using System.Threading.Tasks;
using AutoMapper;
using HslCommunication;
using HslCommunication.WebSocket;
using OfficeOpenXml.FormulaParsing.Excel.Functions.Information;
using SqlSugar;
using WIDESEAWCS_BasicInfoRepository;
using WIDESEAWCS_BasicInfoService;
using WIDESEAWCS_Common;
using WIDESEAWCS_Core;
using WIDESEAWCS_Core.BaseRepository;
using WIDESEAWCS_Core.BaseServices;
using WIDESEAWCS_Core.Enums;
using WIDESEAWCS_Core.Helper;
using WIDESEAWCS_DTO.BasicInfo;
using WIDESEAWCS_DTO.PlacedBlockDTO;
using WIDESEAWCS_DTO.TaskInfo;
using WIDESEAWCS_IBasicInfoRepository;
using WIDESEAWCS_IBasicInfoService;
using WIDESEAWCS_ITaskInfoRepository;
using WIDESEAWCS_ITaskInfoService;
using WIDESEAWCS_Model.Models;
using WIDESEAWCS_QuartzJob.Repository;
using static Dm.net.buffer.ByteArrayBuffer;

namespace WIDESEAWCS_TaskInfoService
{
    /// <summary>
    /// 任务服务类,提供任务相关的业务逻辑操作
    /// </summary>
    /// <remarks>
    /// 主要功能包括: <br/>
    /// 1. 查询门架设备未执行任务 <br/>
    /// 2. 执行订单块放置测试 <br/>
    /// 3. 创建和管理任务 <br/>
    /// 4. 生成和重新生成任务 <br/>
    /// 5. 处理异常任务 <br/>
    /// 6. 任务完成处理 <br/>
    /// 依赖仓储层进行数据访问,通过工作单元管理事务 <br/>
    /// 使用WebSocket进行实时通知,并包含完整的异常处理机制
    /// </remarks>
    public class TaskService : ServiceBase<Dt_Task, ITaskRepository>, ITaskService
    {
        /// <summary>
        /// 工作单元管理接口实例,用于管理数据库事务和工作单元
        /// </summary>
        private readonly IUnitOfWorkManage _unitOfWorkManage;

        /// <summary>
        /// 用于对象映射的映射器实例
        /// </summary>
        private readonly IMapper _mapper;

        /// <summary>
        /// 容器仓储接口实例,用于容器相关数据操作
        /// </summary>
        private readonly IContainerRepository _containerRepository;

        /// <summary>
        /// 容器物品仓储接口实例(只读)
        /// </summary>
        private readonly IContainerItemRepository _containerItemRepository;

        /// <summary>
        /// WebSocket服务器实例,用于处理WebSocket连接
        /// </summary>
        private readonly WebSocketServer _webSocketServer;

        /// <summary>
        /// 订单详情仓储接口实例(只读)
        /// </summary>
        private readonly IOrderDetailsRepository _orderDetailsRepository;

        /// <summary>
        /// 容器服务接口实例,用于管理依赖注入容器相关操作
        /// </summary>
        private readonly IContainerService _containerService;

        /// <summary>
        /// 订单明细服务接口
        /// </summary>
        private readonly IOrderDetailsService _orderDetailsService;

        /// <summary>
        /// 订单行仓储接口实例,用于操作订单行数据
        /// </summary>
        private readonly IOrderrowsRepository _orderrowsRepository;

        /// <summary>
        /// 订单容器仓储接口实例,用于操作订单容器相关数据
        /// </summary>
        private readonly IOrderContainerRepository _orderContainerRepository;

        public TaskService(ITaskRepository BaseDal, IUnitOfWorkManage unitOfWorkManage, IMapper mapper, IContainerRepository containerRepository, IContainerItemRepository containerItemRepository, WebSocketServer webSocketServer, IOrderDetailsRepository orderDetailsRepository, IContainerService containerService, IOrderDetailsService orderDetailsService, IOrderrowsRepository orderrowsRepository, IOrderContainerRepository orderContainerRepository) : base(BaseDal)
        {
            _unitOfWorkManage = unitOfWorkManage;
            _mapper = mapper;
            _containerRepository = containerRepository;
            _containerItemRepository = containerItemRepository;
            _webSocketServer = webSocketServer;
            _orderDetailsRepository = orderDetailsRepository;
            _containerService = containerService;
            _orderDetailsService = orderDetailsService;
            _orderrowsRepository = orderrowsRepository;
            _orderContainerRepository = orderContainerRepository;
        }

        /// <summary>
        /// 查询指定门架设备未执行的任务
        /// </summary>
        /// <param name="gantryDeviceNo">门架设备编号</param>
        /// <returns>未执行的任务数据,若不存在则返回null</returns>
        public Dt_Task? QueryAGantryUnExecuteTask(string gantryDeviceNo)
        {
            return BaseDal.QueryFirst(x => x.TaskState == (int)TaskStatusEnum.Gantry_New && x.DeviceCode == gantryDeviceNo);
        }

        /// <summary>
        /// 执行订单块放置测试,根据订单行ID获取订单明细并计算放置位置
        /// </summary>
        /// <param name="orderRowId">订单行ID</param>
        /// <returns>包含放置块信息的Web响应内容。成功时返回放置块列表,失败时返回错误信息</returns>
        /// <remarks>
        /// 1. 通过订单行ID查询订单明细数据 <br/>
        /// 2. 对每个有效订单明细计算合适的放置位置 <br/>
        /// 3. 创建容器项和任务记录 <br/>
        /// 4. 通过WebSocket发布放置位置信息 <br/>
        /// 5. 返回所有成功放置的块信息
        /// </remarks>
        public WebResponseContent PlaceBlockTest(int orderRowId)
        {
            try
            {
                List<OrderDetails> orderDetails = _orderDetailsRepository.QueryData(x => x.Orderrowsid == orderRowId);
                if (orderDetails == null || orderDetails.Count == 0)
                {
                    return WebResponseContent.Instance.Error("未找到订单明细信息");
                }

                List<PlacedBlock> placedBlocks = new List<PlacedBlock>();

                string putPosition = "";

                for (int i = 0; i < orderDetails.Count; i++)
                {
                    try
                    {
                        lock (placedBlocks)
                        {
                            int length = Convert.ToInt32(orderDetails[i].Orderdetails_length);
                            int width = Convert.ToInt32(orderDetails[i].Orderdetails_width);
                            int height = Convert.ToInt32(orderDetails[i].Orderdetails_thickness);

                            OrderInfo orderInfo = _orderDetailsService.GetOrderInfoByBarcode(orderDetails[i].Orderdetails_outid);

                            var (flag, taskPosition, message) = _containerService.GetPosition(orderInfo.OrderHeadId, orderInfo.OrderNo, length, width, height);
                            if (flag && taskPosition != null)
                            {
                                if (string.IsNullOrEmpty(putPosition))
                                {
                                    putPosition = taskPosition.PutPosition;
                                }

                                if (putPosition != taskPosition.PutPosition)
                                {
                                    continue;
                                }

                                Dt_OrderContainer orderContainer = _orderContainerRepository.QueryFirst(x => x.OrderNo == orderInfo.OrderNo && x.ContainerCode == taskPosition.PutPosition);

                                Dt_Container putContainer = _containerRepository.QueryFirst(x => x.ContainerCode == taskPosition.PutPosition);
                                if (putContainer == null)
                                {
                                    //todo
                                    throw new Exception("未找到放货位置");
                                }

                                bool isAdd = false;
                                if (orderContainer == null)
                                {
                                    orderContainer = new Dt_OrderContainer()
                                    {
                                        OrderId = orderInfo.OrderHeadId,
                                        ContainerCode = putContainer.ContainerCode,
                                        ContainerId = putContainer.Id,
                                        OrderNo = orderInfo.OrderNo,
                                    };
                                    int containerLength = putContainer.ContainerLength;
                                    int containerWidth = putContainer.ContainerWidth;
                                    int maxValue = Math.Max(length, width);
                                    int minValue = Math.Min(length, width);
                                    if (minValue > putContainer.ContainerWidth)
                                    {
                                        containerLength = maxValue;
                                        containerWidth = minValue;
                                    }

                                    orderContainer.MaxLength = containerLength;
                                    orderContainer.MaxWidth = containerWidth;
                                    isAdd = true;
                                }

                                if (taskPosition != null)
                                {
                                    Dt_ContainerItem dt_ContainerItem = new Dt_ContainerItem()
                                    {
                                        ContainerId = putContainer.Id,
                                        ItemCode = orderInfo.Barcode,
                                        ItemLength = length,
                                        ItemWidth = width,
                                        ItemHeight = height,
                                        ItemPositionX = taskPosition.PositionX,
                                        ItemPositionY = taskPosition.PositionY,
                                        ItemPositionZ = taskPosition.PositionZ,
                                        ItemRelaPositionX = taskPosition.PutPositionX,
                                        ItemRelaPositionY = taskPosition.PutPositionY,
                                        ItemRelaPositionZ = taskPosition.PutPositionZ,
                                        ItemStatus = (int)ItemStatusEnum.Assigned,
                                        ItemName = orderInfo.Barcode
                                    };

                                    Dt_Task dt_Task = new Dt_Task()
                                    {
                                        PalletCode = orderInfo.Barcode,
                                        DeviceCode = putContainer.DeviceCode,
                                        TaskState = (int)TaskStatusEnum.Gantry_Wait,
                                        TaskType = 0,
                                        SourceAddress = "",
                                        TargetAddress = $"F06",
                                        CurrentAddress = $"F06*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}",
                                        NextAddress = $"{putContainer.ContainerCode}*{taskPosition.PutPositionX}*{taskPosition.PutPositionY}*{taskPosition.PutPositionZ}*{taskPosition.PositionR}",
                                        ItemInfo = $"{length}*{width}*{height}",
                                        Grade = 0,
                                    };
                                    putContainer.ContainerStatus = ContainerStatusEnum.NonEmpty.ObjToInt();
                                    _unitOfWorkManage.BeginTran();
                                    BaseDal.AddData(dt_Task);
                                    _containerRepository.UpdateData(putContainer);
                                    _containerItemRepository.AddData(dt_ContainerItem);
                                    if (isAdd)
                                    {
                                        _orderContainerRepository.AddData(orderContainer);
                                    }
                                    _unitOfWorkManage.CommitTran();

                                    PlacedBlock placedBlock = new PlacedBlock(new Point3D(taskPosition.PositionX, taskPosition.PositionY, taskPosition.PositionZ), length > width ? length : width, width > length ? length : width, height);
                                    placedBlocks.Add(placedBlock);

                                    Thread.Sleep(1000);
                                    object obj = new
                                    {
                                        x = taskPosition.PutCenterPositionX - putContainer.ContainerLength / 2,
                                        y = taskPosition.PutCenterPositionY - putContainer.ContainerWidth / 2,
                                        z = taskPosition.PutPositionZ,
                                        length = length > width ? length : width,
                                        width = width > length ? length : width,
                                        height,
                                    };
                                    _webSocketServer.PublishAllClientPayload(obj.Serialize());
                                }
                            }
                        }
                    }
                    catch (Exception ex)
                    {

                    }
                }
                return WebResponseContent.Instance.OK(data: placedBlocks);
            }
            catch (Exception ex)
            {
                return WebResponseContent.Instance.Error(ex.Message);
            }
        }

        /// <summary>
        /// 创建任务并计算物品放置位置
        /// </summary>
        /// <param name="takePosition">取货位置编码</param>
        /// <param name="putPosition">放货位置编码</param>
        /// <param name="deviceCode">设备编码</param>
        /// <param name="length">物品长度</param>
        /// <param name="width">物品宽度</param>
        /// <param name="height">物品高度</param>
        /// <returns>包含操作结果的Web响应内容</returns>
        /// <remarks>
        /// 1. 验证取货/放货位置是否存在 <br/>
        /// 2. 计算物品在容器中的最佳放置位置 <br/>
        /// 3. 创建容器物品记录和任务记录 <br/>
        /// 4. 通过WebSocket通知客户端
        /// </remarks>
        public WebResponseContent CreateTask(string takePosition, string putPosition, string deviceCode, int length, int width, int height)
        {
            try
            {
                Dt_Container takeContainer = _containerRepository.QueryFirst(x => x.ContainerCode == takePosition && x.ContainerType == ContainerTypeEnum.TakeContainer.ObjToInt());
                if (takeContainer == null)
                {
                    return WebResponseContent.Instance.Error("取货位置不存在");
                }

                List<Dt_Container> containers = _containerRepository.QueryData(x => x.DeviceCode == deviceCode && x.ContainerType == ContainerTypeEnum.PutContainer.ObjToInt());

                Dt_Container putContainer = _containerRepository.QueryFirst(x => x.ContainerCode == putPosition && x.ContainerType == ContainerTypeEnum.PutContainer.ObjToInt());
                if (putContainer == null)
                {
                    return WebResponseContent.Instance.Error("放货位置不存在");
                }

                int edge = 0;
                if (putContainer.ContainerNo == containers.Min(x => x.ContainerNo))
                {
                    edge = 1;
                }

                ContainerSize containerSize = new ContainerSize(putContainer.ContainerLength, putContainer.ContainerWidth, putContainer.ContainerHeight);
                List<Dt_ContainerItem> containerItems = _containerItemRepository.QueryData(x => x.ContainerId == putContainer.Id);

                List<PlacedBlock> placedBlocks = containerItems.Select(x => new PlacedBlock(new Point3D(x.ItemPositionX, x.ItemPositionY, x.ItemPositionZ), x.ItemLength > x.ItemLength ? x.ItemLength : x.ItemWidth, x.ItemLength > x.ItemLength ? x.ItemWidth : x.ItemLength, x.ItemHeight)).ToList();


                TaskPosition? taskPosition = _containerService.GetTaskPosition(length, width, height, containerSize, placedBlocks, edge);
                if (taskPosition == null)
                {
                    return WebResponseContent.Instance.Error("未找到合适放置位置");
                }

                object obj = new
                {
                    x = taskPosition.PutCenterPositionX - putContainer.ContainerLength / 2,
                    y = taskPosition.PutCenterPositionY - putContainer.ContainerWidth / 2,
                    z = taskPosition.PutCenterPositionZ,
                    length,
                    width,
                    height,
                };

                _webSocketServer.PublishAllClientPayload(obj.Serialize());

                string code = DateTime.Now.ToString("yyyyMMddHHmmss");

                Dt_ContainerItem dt_ContainerItem = new Dt_ContainerItem()
                {
                    ContainerId = putContainer.Id,
                    ItemCode = code,
                    ItemLength = length,
                    ItemWidth = width,
                    ItemHeight = height,
                    ItemPositionX = taskPosition.PositionX,
                    ItemPositionY = taskPosition.PositionY,
                    ItemPositionZ = taskPosition.PositionZ,
                    ItemRelaPositionX = taskPosition.PutPositionX,
                    ItemRelaPositionY = taskPosition.PutPositionY,
                    ItemRelaPositionZ = taskPosition.PutPositionZ,
                    ItemStatus = (int)ItemStatusEnum.Assigned,
                    ItemName = code
                };

                Dt_Task dt_Task = new Dt_Task()
                {
                    PalletCode = code,
                    DeviceCode = putContainer.DeviceCode,
                    TaskState = (int)TaskStatusEnum.Gantry_New,
                    TaskType = 0,
                    SourceAddress = "",
                    TargetAddress = $"{takeContainer.ContainerNo}",
                    CurrentAddress = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}",
                    NextAddress = $"{putContainer.ContainerCode}*{taskPosition.PutPositionX}*{taskPosition.PutPositionY}*{taskPosition.PutPositionZ}*{taskPosition.PositionR}",
                    ItemInfo = $"{length}*{width}*{height}",
                    Grade = 0,
                };

                _containerItemRepository.AddData(dt_ContainerItem);
                base.AddData(dt_Task);

                return WebResponseContent.Instance.OK();
            }
            catch (Exception ex)
            {
                return WebResponseContent.Instance.Error(ex.Message);
            }
        }

        /// <summary>
        /// 根据订单信息生成任务
        /// </summary>
        /// <param name="orderInfo">订单信息</param>
        /// <returns>包含三个值的元组: <br/>
        /// Item1 - 操作是否成功 (bool) <br/>
        /// Item2 - 生成的任务对象 (Dt_Task),失败时为null <br/>
        /// Item3 - 错误消息 (string),成功时为空字符串
        /// </returns>
        /// <remarks>
        /// 该方法会: <br/>
        /// 1. 根据订单尺寸获取合适的货位 <br/>
        /// 2. 创建容器项和任务记录 <br/>
        /// 3. 更新相关容器状态 <br/>
        /// 4. 在事务中执行所有数据库操作
        /// </remarks>
        public (bool, Dt_Task?, string) GenerateTask(OrderInfo orderInfo)
        {
            try
            {
                int length = Convert.ToInt32(orderInfo.Length);
                int width = Convert.ToInt32(orderInfo.Width);
                int height = Convert.ToInt32(orderInfo.Thickness);

                var (flag, taskPosition, message) = _containerService.GetPosition(orderInfo.OrderHeadId, orderInfo.OrderNo, length, width, height);
                if (flag && taskPosition != null)
                {
                    Dt_Container putContainer = _containerRepository.QueryFirst(x => x.ContainerCode == taskPosition.PutPosition);
                    if (putContainer == null)
                    {
                        return (false, null, "未找到放货位置");
                    }

                    Dt_Container? takeContainer = GetTakePosition(putContainer.DeviceCode);
                    if (takeContainer == null)
                    {
                        return (false, null, "未找到取货位置");
                    }

                    Dt_OrderContainer orderContainer = _orderContainerRepository.QueryFirst(x => x.OrderNo == orderInfo.OrderNo && x.ContainerCode == putContainer.ContainerCode);

                    bool isAdd = false;
                    bool isUpdate = false;
                    if (orderContainer == null)
                    {
                        orderContainer = new Dt_OrderContainer()
                        {
                            OrderId = orderInfo.OrderHeadId,
                            ContainerCode = putContainer.ContainerCode,
                            ContainerId = putContainer.Id,
                            OrderNo = orderInfo.OrderNo,
                        };

                        int containerLength = putContainer.ContainerLength;
                        int containerWidth = putContainer.ContainerWidth;
                        int maxValue = Math.Max(length, width);
                        int minValue = Math.Min(length, width);
                        if (minValue > putContainer.ContainerWidth)
                        {
                            containerLength = maxValue;
                            containerWidth = minValue;
                        }

                        orderContainer.MaxLength = containerLength;
                        orderContainer.MaxWidth = containerWidth;
                        isAdd = true;
                    }

                    Dt_ContainerItem dt_ContainerItem = new Dt_ContainerItem()
                    {
                        ContainerId = putContainer.Id,
                        ItemCode = orderInfo.Barcode,
                        ItemLength = length,
                        ItemWidth = width,
                        ItemHeight = height,
                        ItemPositionX = taskPosition.PositionX,
                        ItemPositionY = taskPosition.PositionY,
                        ItemPositionZ = taskPosition.PositionZ,
                        ItemRelaPositionX = taskPosition.PutPositionX,
                        ItemRelaPositionY = taskPosition.PutPositionY,
                        ItemRelaPositionZ = taskPosition.PutPositionZ,
                        ItemStatus = (int)ItemStatusEnum.Assigned,
                        ItemName = orderInfo.Barcode,
                        Remark = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}"
                    };

                    Dt_Task dt_Task = new Dt_Task()
                    {
                        PalletCode = orderInfo.Barcode,
                        DeviceCode = putContainer.DeviceCode,
                        TaskState = (int)TaskStatusEnum.Gantry_Wait,
                        TaskType = 0,
                        SourceAddress = "",
                        TargetAddress = $"{takeContainer.ContainerNo}",
                        CurrentAddress = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}",
                        NextAddress = $"{putContainer.ContainerCode}*{taskPosition.PutPositionX}*{taskPosition.PutPositionY}*{taskPosition.PutPositionZ}*{taskPosition.PositionR}",
                        ItemInfo = $"{length}*{width}*{height}",
                        Grade = taskPosition.Edge
                    };
                    putContainer.ContainerStatus = ContainerStatusEnum.NonEmpty.ObjToInt();
                    _unitOfWorkManage.BeginTran();
                    BaseDal.AddData(dt_Task);
                    _containerRepository.UpdateData(putContainer);
                    _containerItemRepository.AddData(dt_ContainerItem);
                    if (isAdd)
                    {
                        _orderContainerRepository.AddData(orderContainer);
                    }
                    else if (isUpdate)
                    {
                        _orderContainerRepository.UpdateData(orderContainer);
                    }
                    _unitOfWorkManage.CommitTran();

                    return (true, dt_Task, "");
                }
                else
                {
                    return (false, null, message);
                }
            }
            catch (Exception ex)
            {
                _unitOfWorkManage.RollbackTran();
                return (false, null, $"错误,{ex.ExceptionToString()}");
            }
        }

        /// <summary>
        /// 重新生成任务
        /// </summary>
        /// <param name="task">原始任务对象</param>
        /// <param name="stationCode">工作站编码</param>
        /// <returns>
        /// 包含三个值的元组: <br/>
        /// 1. bool - 操作是否成功 <br/>
        /// 2. Dt_Task - 更新后的任务对象(失败时为null) <br/>
        /// 3. string - 错误消息(成功时为空字符串)
        /// </returns>
        /// <remarks>
        /// 该方法用于重新生成任务信息,包括: <br/>
        /// 1. 解析板子尺寸信息 <br/>
        /// 2. 验证取货/放货位置 <br/>
        /// 3. 计算物品放置位置 <br/>
        /// 4. 更新任务状态和位置信息 <br/>
        /// 5. 在事务中更新数据库记录
        /// </remarks>
        public (bool, Dt_Task?, string) RegenerateTask(Dt_Task task, string stationCode)
        {
            try
            {
                int length = 0;
                int width = 0;
                int height = 0;
                Dt_ContainerItem containerItem = _containerItemRepository.QueryFirst(x => x.ItemCode == task.PalletCode);
                if (containerItem == null)
                {
                    List<string> itemInfos = task.ItemInfo.Split("*").ToList();
                    if (itemInfos.Count == 3)
                    {
                        length = Convert.ToInt32(itemInfos[0]);
                        width = Convert.ToInt32(itemInfos[1]);
                        height = Convert.ToInt32(itemInfos[2]);
                    }
                    else
                    {
                        return (false, null, "板子尺寸信息错误");
                    }
                }
                else
                {
                    length = containerItem.ItemLength;
                    width = containerItem.ItemWidth;
                    height = containerItem.ItemHeight;
                }
                string containerCode = task.NextAddress.Split("*")[0];

                Dt_Container putContainer = _containerRepository.QueryFirst(x => x.ContainerCode == containerCode);

                if (putContainer == null)
                {
                    return (false, null, "未找到放货位置");
                }

                Dt_Container? takeContainer = _containerRepository.QueryFirst(x => x.ContainerCode == stationCode);
                if (takeContainer == null)
                {
                    return (false, null, "未找到取货位置");
                }

                var (flag, taskPosition, message) = _containerService.GetPosition(putContainer, length, width, height);
                if (!flag || taskPosition == null)
                {
                    return (false, null, message);
                }

                Dt_OrderContainer orderContainer = _orderContainerRepository.QueryFirst(x => x.ContainerCode == putContainer.ContainerCode);

                bool isAdd = false;

                if (orderContainer == null)
                {
                    return (false, null, "未找到订单与容器绑定关系");
                }

                int containerLength = putContainer.ContainerLength;
                int containerWidth = putContainer.ContainerWidth;
                int maxValue = Math.Max(length, width);
                int minValue = Math.Min(length, width);
                if (minValue > putContainer.ContainerWidth)
                {
                    containerLength = maxValue;
                    containerWidth = minValue;
                }

                orderContainer.MaxLength = containerLength;
                orderContainer.MaxWidth = containerWidth;

                if (containerItem == null)
                {
                    containerItem = new Dt_ContainerItem()
                    {
                        ContainerId = putContainer.Id,
                        ItemCode = task.PalletCode,
                        ItemLength = length,
                        ItemWidth = width,
                        ItemHeight = height,
                        ItemPositionX = taskPosition.PositionX,
                        ItemPositionY = taskPosition.PositionY,
                        ItemPositionZ = taskPosition.PositionZ,
                        ItemRelaPositionX = taskPosition.PutPositionX,
                        ItemRelaPositionY = taskPosition.PutPositionY,
                        ItemRelaPositionZ = taskPosition.PutPositionZ,
                        ItemStatus = (int)ItemStatusEnum.Assigned,
                        ItemName = task.PalletCode,
                        Remark = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}"
                    };
                    isAdd = true;
                }
                else
                {
                    containerItem.ItemLength = length;
                    containerItem.ItemWidth = width;
                    containerItem.ItemHeight = height;
                    containerItem.ItemPositionX = taskPosition.PositionX;
                    containerItem.ItemPositionY = taskPosition.PositionY;
                    containerItem.ItemPositionZ = taskPosition.PositionZ;
                    containerItem.ItemRelaPositionX = taskPosition.PutPositionX;
                    containerItem.ItemRelaPositionY = taskPosition.PutPositionY;
                    containerItem.ItemRelaPositionZ = taskPosition.PutPositionZ;
                    containerItem.ItemStatus = (int)ItemStatusEnum.Assigned;
                    containerItem.Remark = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}";
                }

                task.TaskState = (int)TaskStatusEnum.Gantry_New;
                task.CurrentAddress = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}";
                task.NextAddress = $"{putContainer.ContainerCode}*{taskPosition.PutPositionX}*{taskPosition.PutPositionY}*{taskPosition.PutPositionZ}*{taskPosition.PositionR}";
                task.ItemInfo = $"{length}*{width}*{height}";

                _unitOfWorkManage.BeginTran();
                BaseDal.UpdateData(task);
                _containerRepository.UpdateData(putContainer);
                _orderContainerRepository.UpdateData(orderContainer);

                if (isAdd)
                {
                    _containerItemRepository.AddData(containerItem);
                }
                else
                {
                    _containerItemRepository.UpdateData(containerItem);
                }
                _unitOfWorkManage.CommitTran();

                return (true, task, "");

            }
            catch (Exception ex)
            {
                _unitOfWorkManage.RollbackTran();
                return (false, null, $"错误,{ex.ExceptionToString()}");
            }
        }

        /// <summary>
        /// 生成异常任务
        /// </summary>
        /// <param name="orderInfo">订单信息</param>
        /// <returns>元组包含三个值: <br/>
        /// Item1 - 操作是否成功(bool) <br/>
        /// Item2 - 生成的任务对象(Dt_Task),失败时为null <br/>
        /// Item3 - 错误消息(string),成功时为空字符串</returns>
        /// <remarks>
        /// 该方法会根据订单信息生成一个异常处理任务,包括: <br/>
        /// 1. 获取异常位置 <br/>
        /// 2. 验证放货/取货位置 <br/>
        /// 3. 创建容器项和任务记录 <br/>
        /// 4. 提交事务
        /// </remarks>
        public (bool, Dt_Task?, string) GenerateExceptionTask(OrderInfo orderInfo)
        {
            int length = Convert.ToInt32(orderInfo.Length);
            int width = Convert.ToInt32(orderInfo.Width);
            int height = Convert.ToInt32(orderInfo.Thickness);

            var (flag, taskPosition, message) = _containerService.GetExceptionPosition(length, width, height);
            if (flag && taskPosition != null)
            {
                Dt_Container putContainer = _containerRepository.QueryFirst(x => x.ContainerCode == taskPosition.PutPosition);
                if (putContainer == null)
                {
                    return (false, null, "未找到放货位置");
                }

                Dt_Container? takeContainer = GetTakePosition(putContainer.DeviceCode);
                if (takeContainer == null)
                {
                    return (false, null, "未找到取货位置");
                }

                Dt_ContainerItem dt_ContainerItem = new Dt_ContainerItem()
                {
                    ContainerId = putContainer.Id,
                    ItemCode = orderInfo.Barcode,
                    ItemLength = length,
                    ItemWidth = width,
                    ItemHeight = height,
                    ItemPositionX = taskPosition.PositionX,
                    ItemPositionY = taskPosition.PositionY,
                    ItemPositionZ = taskPosition.PositionZ,
                    ItemRelaPositionX = taskPosition.PutPositionX,
                    ItemRelaPositionY = taskPosition.PutPositionY,
                    ItemRelaPositionZ = taskPosition.PutPositionZ,
                    ItemStatus = (int)ItemStatusEnum.Assigned,
                    ItemName = orderInfo.Barcode,
                    Remark = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}"
                };

                Dt_Task dt_Task = new Dt_Task()
                {
                    PalletCode = orderInfo.Barcode,
                    DeviceCode = putContainer.DeviceCode,
                    TaskState = (int)TaskStatusEnum.Gantry_Wait,
                    TaskType = 0,
                    SourceAddress = "",
                    TargetAddress = $"{takeContainer.ContainerNo}",
                    CurrentAddress = $"{takeContainer.ContainerCode}*{taskPosition.TakePositionX}*{taskPosition.TakePositionY}*{taskPosition.TakePositionZ}*{taskPosition.PositionR}",
                    NextAddress = $"{putContainer.ContainerCode}*{taskPosition.PutPositionX}*{taskPosition.PutPositionY}*{taskPosition.PutPositionZ}*{taskPosition.PositionR}",
                    ItemInfo = $"{length}*{width}*{height}",
                    Grade = 0,
                };

                if (taskPosition.TakeCenterPositionZ == 99)
                {
                    dt_Task.TaskState = (int)TaskStatusEnum.Gantry_BeRelease;
                }

                _unitOfWorkManage.BeginTran();
                BaseDal.AddData(dt_Task);
                _containerItemRepository.AddData(dt_ContainerItem);
                _unitOfWorkManage.CommitTran();

                return (true, dt_Task, "");
            }
            else
            {
                return (false, null, message);
            }
        }

        /// <summary>
        /// 根据设备编号获取取货位置
        /// </summary>
        /// <param name="deviceCode">设备编号</param>
        /// <returns>符合条件的容器对象,若找不到则返回null</returns>
        /// <remarks>
        /// 1. 首先查询指定设备下所有可用的取货容器,并按ContainerSort升序排序 <br/>
        /// 2. 查找该设备最近的任务记录 <br/>
        /// 3. 如果找到任务记录,则根据任务源地址匹配容器,返回下一个容器(若已是最后一个则返回第一个) <br/>
        /// 4. 如果出现异常或未找到匹配项,则返回第一个可用容器
        /// </remarks>
        public Dt_Container? GetTakePosition(string deviceCode)
        {
            Dictionary<string, OrderByType> orderby = new Dictionary<string, OrderByType>() { { nameof(Dt_Container.ContainerSort), OrderByType.Asc } };
            List<Dt_Container> containers = _containerRepository.QueryData(x => x.ContainerType == ContainerTypeEnum.TakeContainer.ObjToInt() && x.DeviceCode == deviceCode && x.ContainerEnable, orderby);

            try
            {
                if (containers == null || containers.Count == 0)
                {
                    throw new Exception($"未找到对应的取货位置,设备编号:{deviceCode}");
                }

                Dictionary<string, OrderByType> taskOrderby = new Dictionary<string, OrderByType>() { { nameof(Dt_Task.TaskNum), OrderByType.Desc } };
                Dt_Task task = BaseDal.QueryFirst(x => x.DeviceCode == deviceCode, taskOrderby);
                if (task != null)
                {
                    string? sourceCode = task.SourceAddress.Split("*").FirstOrDefault();
                    if (!string.IsNullOrEmpty(sourceCode))
                    {
                        Dt_Container? container = containers.FirstOrDefault(x => x.ContainerCode == sourceCode);
                        if (container != null)
                        {
                            int index = containers.IndexOf(container);

                            if (index + 1 < containers.Count)
                            {
                                return containers[index + 1];
                            }
                            else
                            {
                                return containers.FirstOrDefault();
                            }
                        }
                    }
                }
            }
            catch (Exception ex)
            {

            }
            return containers.FirstOrDefault();
        }

        /// <summary>
        /// 任务完成处理方法
        /// </summary>
        /// <param name="task">任务实体</param>
        /// <returns>Web响应内容</returns>
        /// <remarks>
        /// 1. 更新任务状态为"龙门架完成" <br/>
        /// 2. 更新相关订单详情状态为"码垛成功" <br/>
        /// 3. 更新订单行中的托盘数量 <br/>
        /// 4. 根据任务完成情况发送WebSocket通知 <br/>
        /// 5. 记录操作日志并提交事务 <br/>
        /// 6. 异常时回滚事务并返回错误信息
        /// </remarks>
        public WebResponseContent TaskComplete(Dt_Task task)
        {
            try
            {
                task.TaskState = TaskStatusEnum.Gantry_Completed.ObjToInt();
                _unitOfWorkManage.BeginTran();
                OrderDetails orderDetails = _orderDetailsRepository.QueryFirst(x => x.Orderdetails_outid == task.PalletCode);
                if (orderDetails != null)
                {
                    orderDetails.Orderdetails_status = PalletingStatusEnmu.PalletingSuccess.ObjToInt();

                    Orderrows orderrows = _orderrowsRepository.QueryFirst(x => x.id == orderDetails.Orderrowsid);
                    if (orderrows != null)
                    {
                        orderrows.Orderrows_PalletNum = orderrows.Orderrows_PalletNum ?? 0 + 1;

                        Dt_Container container = _containerRepository.QueryFirst(x => task.NextAddress.Contains(x.ContainerCode));

                        _orderDetailsRepository.UpdateData(orderDetails);
                        _orderrowsRepository.UpdateData(orderrows);

                        if (container.ContainerType == ContainerTypeEnum.ExceptionContainer.ObjToInt())
                        {
                            int count = Db.Queryable<Dt_ContainerItem>().Count(x => x.ContainerId == container.Id);
                            if (count >= 5)
                            {
                                if (LightStatusStorage.LightStatusDic.ContainsKey(container.ContainerCode))
                                {
                                    LightStatusStorage.LightStatusDic[container.ContainerCode] = LightStatusEnum.LightError;
                                }
                            }
                        }

                        try
                        {
                            List<string> containerItemCodes = _containerItemRepository.QueryData(x => x.ContainerId == container.Id).Select(x => x.ItemCode).ToList();

                            List<OrderDetails> totalDetails = _orderDetailsRepository.QueryData(x => x.Orderrowsid == orderrows.id);
                            List<OrderDetails> details = totalDetails.Where(x => x.Orderrowsid == orderrows.id && x.Orderdetails_status == PalletingStatusEnmu.PalletingSuccess.ObjToInt() && containerItemCodes.Contains(x.Orderdetails_outid)).ToList();

                            int sortedNum = totalDetails.Where(x => x.Orderrowsid == orderrows.id && x.Orderdetails_status == PalletingStatusEnmu.PalletingSuccess.ObjToInt()).Count();

                            List<object> orderData = new List<object>();
                            foreach (var item in details)
                            {
                                object obj = new
                                {
                                    name = item.Orderdetails_name,
                                    baseName = item.Orderdetails_productName,
                                    size = $"{item.Orderdetails_length}*{item.Orderdetails_width}*{item.Orderdetails_thickness}",
                                    process = "",
                                };
                                orderData.Add(obj);
                            }
                            object data = new
                            {
                                orderCode = orderrows.Orderrows_orderid,
                                orderName = orderrows.Orderrows_name,
                                cusName = orderrows.Orderrows_customer,
                                stationCode = container.ContainerCode,
                                orderTotalNum = totalDetails.Count,
                                sortedNum = sortedNum,
                                unsortedNum = totalDetails.Count - sortedNum,
                                stationSortedNum = details.Count,
                                orderData = orderData,
                                orderId = orderrows.id,
                                release = 0
                            };


                            _webSocketServer.PublishAllClientPayload(data.Serialize());

                        }
                        catch (Exception ex)
                        {

                        }
                    }
                }

                BaseDal.DeleteAndMoveIntoHty(task, App.User?.UserId > 0 ? OperateTypeEnum.人工完成 : OperateTypeEnum.自动完成);
                _unitOfWorkManage.CommitTran();
                return WebResponseContent.Instance.OK();
            }
            catch (Exception ex)
            {
                _unitOfWorkManage.RollbackTran();
                return WebResponseContent.Instance.Error(ex.Message);
            }
        }
    }
}