using AutoMapper;
|
using SqlSugar;
|
using WIDESEA_Common.LocationEnum;
|
using WIDESEA_Common.StockEnum;
|
using WIDESEA_Core.BaseRepository;
|
using WIDESEA_Core.BaseServices;
|
using WIDESEA_Core.Helper;
|
using WIDESEA_IBasicService;
|
using WIDESEA_IRecordService;
|
using WIDESEA_IStockService;
|
using WIDESEA_Model.Models;
|
|
namespace WIDESEA_StockService
|
{
|
public partial class StockInfoService : ServiceBase<Dt_StockInfo, IRepository<Dt_StockInfo>>, IStockInfoService
|
{
|
private readonly IMapper _mapper;
|
|
private readonly IRecordService _recordService;
|
public IRepository<Dt_StockInfo> Repository => BaseDal;
|
private readonly IRepository<Dt_StockInfoDetail> _stockInfoDetailRepository;
|
|
private readonly ILocationInfoService _locationInfoService;
|
public StockInfoService(IRepository<Dt_StockInfo> BaseDal, IMapper mapper, IRepository<Dt_StockInfoDetail> stockInfoDetailRepository, IRecordService recordService, ILocationInfoService locationInfoService) : base(BaseDal)
|
{
|
_mapper = mapper;
|
_stockInfoDetailRepository = stockInfoDetailRepository;
|
_recordService = recordService;
|
_locationInfoService = locationInfoService;
|
}
|
|
/// <summary>
|
/// 根据托盘号查询库存
|
/// </summary>
|
/// <param name="palletCode"></param>
|
/// <returns></returns>
|
public Dt_StockInfo? GetStockByPalletCode(string palletCode)
|
{
|
Dt_StockInfo stockInfo = BaseDal.QueryFirst(x => x.PalletCode == palletCode);
|
if (stockInfo != null)
|
{
|
stockInfo.Details = _stockInfoDetailRepository.QueryData(x => x.StockId == stockInfo.Id);
|
}
|
return stockInfo;
|
}
|
|
|
public void AddMaterielGroup(Dt_StockInfo stockInfo)
|
{
|
decimal beforeQuantity = 0;
|
List<Dt_StockInfoDetail> details = new List<Dt_StockInfoDetail>();
|
if (stockInfo.Id == 0)
|
{
|
if (stockInfo.Details != null && stockInfo.Details.Any())
|
{
|
BaseDal.Db.InsertNav(stockInfo).Include(x => x.Details).ExecuteCommand();
|
}
|
else
|
{
|
BaseDal.AddData(stockInfo);
|
}
|
details = stockInfo.Details;
|
}
|
else
|
{
|
beforeQuantity = stockInfo.Details.Where(x => x.Id != 0).Sum(x => x.StockQuantity);
|
|
for (int i = 0; i < stockInfo.Details.Count; i++)
|
{
|
if (stockInfo.Details[i].Id == 0)
|
{
|
details.Add(_stockInfoDetailRepository.Db.Insertable(stockInfo.Details[i]).ExecuteReturnEntity());
|
}
|
|
}
|
|
}
|
stockInfo.Details = details;
|
_recordService.StockQuantityChangeRecordService.AddStockChangeRecord(stockInfo, stockInfo.Details, beforeQuantity, stockInfo.Details.Sum(x => x.StockQuantity) + beforeQuantity, WIDESEA_Common.StockEnum.StockChangeType.MaterielGroup);
|
}
|
|
/// <summary>
|
///
|
/// </summary>
|
/// <param name="stockInfos"></param>
|
/// <param name="materielCode"></param>
|
/// <param name="needQuantity"></param>
|
/// <param name="residueQuantity"></param>
|
/// <returns></returns>
|
public (List<Dt_StockInfo>, Dictionary<int, decimal>) GetOutboundStocks(List<Dt_StockInfo> stockInfos, string materielCode, decimal needQuantity, out decimal residueQuantity)
|
{
|
List<Dt_StockInfo> outStocks = new List<Dt_StockInfo>();
|
Dictionary<int, decimal> stockAllocations = new Dictionary<int, decimal>(); // 记录每个库存明细的分配数量
|
|
// 按先进先出排序所有相关的库存明细
|
var sortedStockDetails = stockInfos
|
.SelectMany(x => x.Details)
|
.Where(x => x.MaterielCode == materielCode &&
|
x.StockQuantity > x.OutboundQuantity) // 有可用库存
|
.OrderBy(x => x.ProductionDate) // 按生产日期排序,先进先出
|
.ThenBy(x => x.StockId) // 相同生产日期按库存ID排序
|
.ToList();
|
|
if (!sortedStockDetails.Any())
|
{
|
residueQuantity = needQuantity;
|
return (outStocks, stockAllocations);
|
}
|
|
// 计算总可用库存
|
var stockTotalQuantity = sortedStockDetails.Sum(x => x.StockQuantity - x.OutboundQuantity);
|
|
if (stockTotalQuantity < needQuantity)
|
{
|
residueQuantity = needQuantity - stockTotalQuantity;
|
// 不抛出异常,允许部分分配
|
}
|
else
|
{
|
residueQuantity = 0;
|
}
|
|
decimal remainingNeed = needQuantity;
|
|
// 按先进先出顺序分配库存
|
foreach (var detail in sortedStockDetails)
|
{
|
if (remainingNeed <= 0) break;
|
|
decimal availableQuantity = detail.StockQuantity - detail.OutboundQuantity;
|
if (availableQuantity <= 0) continue;
|
|
decimal allocateQuantity = Math.Min(availableQuantity, remainingNeed);
|
|
// 更新出库数量
|
detail.OutboundQuantity += allocateQuantity;
|
remainingNeed -= allocateQuantity;
|
|
// 记录分配数量
|
stockAllocations[detail.Id] = allocateQuantity;
|
|
// 如果这个库存还没添加到出库列表中,就添加
|
var stockInfo = stockInfos.First(x => x.Id == detail.StockId);
|
if (!outStocks.Contains(stockInfo))
|
{
|
outStocks.Add(stockInfo);
|
}
|
}
|
|
residueQuantity = remainingNeed;
|
return (outStocks, stockAllocations);
|
}
|
|
/// <summary>
|
/// 根据条码获取库存信息
|
/// </summary>
|
public async Task<Dt_StockInfoDetail> GetStockDetailByBarcode(string barcode, string materielCode)
|
{
|
return await Db.Queryable<Dt_StockInfoDetail>()
|
.Includes(x => x.StockInfo)
|
.Where(x => x.Barcode == barcode && x.MaterielCode == materielCode)
|
.FirstAsync();
|
}
|
|
/// <summary>
|
/// 获取物料的所有条码信息
|
/// </summary>
|
public async Task<List<Dt_StockInfoDetail>> GetMaterialBarcodes(string materielCode, string batchNo = null)
|
{
|
var query = Db.Queryable<Dt_StockInfoDetail>()
|
.Includes(x => x.StockInfo)
|
.Where(x => x.MaterielCode == materielCode && x.StockQuantity > x.OutboundQuantity);
|
|
if (!string.IsNullOrEmpty(batchNo))
|
{
|
query = query.Where(x => x.BatchNo == batchNo);
|
}
|
|
return await query.OrderBy(x => x.ProductionDate).ToListAsync();
|
}
|
public List<Dt_StockInfo> GetStockInfos(string materielCode, string lotNo, List<string> locationCodes)
|
{
|
var query = Db.Queryable<Dt_StockInfo>()
|
.Where(x => locationCodes.Contains(x.LocationCode)
|
// && x.StockStatus == (int)StockStatusEmun.正常)
|
).Includes(x => x.Details);
|
|
if (!string.IsNullOrEmpty(materielCode))
|
{
|
query = query.Where(x => x.Details.Any(d => d.MaterielCode == materielCode));
|
}
|
|
if (!string.IsNullOrEmpty(lotNo))
|
{
|
query = query.Where(x => x.Details.Any(d => d.BatchNo == lotNo));
|
}
|
|
var stocks = query.ToList();
|
|
return stocks.OrderBy(x => x.Details.Where(d => d.MaterielCode == materielCode &&
|
(string.IsNullOrEmpty(lotNo) || d.BatchNo == lotNo)).Min(d => d.ProductionDate)).ToList();
|
//ISugarQueryable<Dt_LocationInfo> sugarQueryable = Db.Queryable<Dt_LocationInfo>().Where(x => locationCodes.Contains(x.LocationCode));
|
//ISugarQueryable<Dt_StockInfo> sugarQueryable1 = Db.Queryable<Dt_StockInfo>().Includes(x => x.Details).Where(x => x.Details.Any(v => v.MaterielCode == materielCode));
|
//return sugarQueryable.InnerJoin(sugarQueryable1, (a, b) => a.LocationCode == b.LocationCode).Select((a, b) => b).OrderBy(a => a.CreateDate).Includes(a => a.Details).ToList();
|
}
|
|
public List<Dt_StockInfo> GetUseableStocks(string materielCode, string batchNo)
|
{
|
List<string> locationCodes = _locationInfoService.GetCanOutLocationCodes();
|
|
return GetStockInfos(materielCode, batchNo, locationCodes);
|
}
|
|
public Dt_StockInfo GetPalletStockInfo(int locationType)
|
{
|
|
return Db.Queryable<Dt_StockInfo>().Where(x => x.StockStatus == StockStatusEmun.入库完成.ObjToInt() && SqlFunc.Subqueryable<Dt_LocationInfo>().Where(v => v.LocationCode == x.LocationCode && v.LocationType == locationType && v.LocationStatus == LocationStatusEnum.Pallet.ObjToInt() && (v.EnableStatus == EnableStatusEnum.OnlyOut.ObjToInt() || EnableStatusEnum.Normal.ObjToInt() == v.EnableStatus)).Any()).OrderBy(x => x.ModifyDate).First();
|
}
|
}
|
}
|