Files
Annamaria Sassi 2b4cad4234 - Aggiunte pagine: clienti e fornitori
- Inserito filtro in offerte in base allo stato
2026-05-15 09:30:38 +02:00

636 lines
22 KiB
C#

using EgwCoreLib.Lux.Core.Generic;
using EgwCoreLib.Razor;
using static EgwCoreLib.Lux.Core.Enums;
namespace Lux.UI.Components.Pages
{
public partial class Offers
{
#region Protected Properties
/// <summary>
/// Filtro offerte: ogni stato / sole aperte
/// </summary>
protected bool AllStates
{
get => allState;
set
{
if (allState != value)
{
allState = value;
DoFilter();
UpdateTable();
}
}
}
private List<string> OfferStateList
{
get => Enum.GetNames(typeof(Enums.OfferStates))
.ToList();
}
private string searchValByState { get; set; } = Enums.OfferStates.Open.ToString();
private string SearchVal
{
get => searchValByState;
set
{
if (searchValByState != value)
{
searchValByState = value;
DoFilter();
UpdateTable();
}
}
}
#endregion Protected Properties
#region Protected Methods
protected override async Task OnInitializedAsync()
{
// prendo il trim corrente + anno precedente...
PeriodoSel = new DtUtils.Periodo(DtUtils.PeriodSet.ThisTrim);
PeriodoSel.Inizio = PeriodoSel.Inizio.AddYears(-1);
SetupArrows();
ConfInit();
await ReloadBaseData();
await ReloadDataAsync();
UpdateTable();
}
#endregion Protected Methods
#region Private Fields
private List<EnvirParamModel> AllConfEnvir = new();
private List<OfferModel> AllRecords = new List<OfferModel>();
private bool allState = false;
/// <summary>
/// Base path x network share files
/// </summary>
private string basePath = "unsafe_uploads";
private int currPage = 1;
private CompileStep currStep = CompileStep.Draft;
private OfferModel? EditRecord = null;
private OfferModel? EditStateRec = null;
private bool isLoading = false;
private List<OfferModel> ListFilt = new List<OfferModel>();
private List<OfferModel> ListRecords = new List<OfferModel>();
private List<CompileStep> ListtChanges = new();
private int numRecord = 10;
/// <summary>
/// Periodo selezionato attuale
/// </summary>
private DtUtils.Periodo PeriodoSel = new DtUtils.Periodo(DtUtils.PeriodSet.ThisYear);
private string searchVal = "";
private OfferModel? SelRecord = null;
private int totalCount = 0;
private string txtStyle = "font-size: 1.2em; font-weight:bold; fill: white;";
private BootstrapModal Modal = new();
private string mTitle = "";
private string mMessage = "";
private BootstrapModal.ModalMode mMode = BootstrapModal.ModalMode.ND;
private Dictionary<bool, string> modalOpt = new Dictionary<bool, string>();
#endregion Private Fields
#region Private Properties
[Inject]
private IConfiguration Config { get; set; } = null!;
[Inject]
private ICalcRuidService CRService { get; set; } = null!;
private string DivMainCss
{
get => SelRecord != null ? "col-6" : "col-12";
}
[Inject]
private IDataLayerServices DLService { get; set; } = null!;
[Inject]
private IEnvirParamService EPService { get; set; } = null!;
[Inject]
private IJSRuntime JSRuntime { get; set; } = null!;
private List<string> listBord01 { get; set; } = new();
[Inject]
private IOfferService OffService { get; set; } = default!;
[Inject]
private IOrderService OrdService { get; set; } = default!;
[Inject]
private IProductionItemService PIService { get; set; } = default!;
[Inject]
private IProdService PService { get; set; } = null!;
private string txtState
{
get => allState ? "Tutte" : "Aperte";
}
private string apiUrl = "";
#endregion Private Properties
#region Private Methods
private void AdvStep(CompileStep newStep)
{
currStep = newStep;
}
private string ArrowBackCol(CompileStep arrowStep)
{
string answ = $"fill: #000000;";
if (arrowStep == currStep)
{
answ = $"fill: #123456;";
}
else if (arrowStep < currStep)
{
answ = $"fill: #456789;";
}
else
{
answ = $"fill: #89ABCD;";
}
return answ;
}
private string CheckSelect(OfferModel curRec)
{
string answ = "";
if (SelRecord != null)
{
answ = curRec.OfferID == SelRecord.OfferID ? "table-info" : "";
}
return answ;
}
private void ConfInit()
{
basePath = Config.GetValue<string>("ServerConf:FileSharePath") ?? "unsafe_uploads";
apiUrl = Config.GetValue<string>("ServerConf:ReportUrl") ?? "";
}
private void DoAction(string args)
{
if (args == "EditRow")
{
currStep = CompileStep.Rows;
EditRecord = SelRecord;
}
}
private void DoAdd()
{
EditRecord = new OfferModel()
{
RefYear = DateTime.Today.Year,
Description = $"Nuova Offerta {DateTime.Today:ddd yyyy.MM.dd}",
ValidUntil = DateTime.Today.AddMonths(1)
};
}
private async Task DoClone(OfferModel rec2clone)
{
mTitle = "Attenzione";
mMessage = "Confermi di voler duplicare interamente l'offerta selezionata?";
mMode = BootstrapModal.ModalMode.Confirm;
modalOpt = new();
modalOpt.Add(true, "Si");
modalOpt.Add(false, "No");
if (!await Modal!.ShowAsync<bool>())
return;
// clona intera offerta + tutte le righe...
await OffService.CloneAsync(rec2clone);
await ReloadDataAsync();
UpdateTable();
}
private async Task DoCancel()
{
int prevId = EditRecord?.OfferID ?? 0;
EditRecord = null;
await ReloadDataAsync();
await Task.Delay(10);
EditRecord = ListRecords.FirstOrDefault(x => x.OfferID == prevId);
}
private void DoEdit(OfferModel curRec)
{
currStep = CompileStep.Header;
EditRecord = curRec;
}
private async Task DoEsc()
{
// chiudo offerta se confermo
mTitle = "Chiusura";
mMessage = "Confermi di voler uscire dall'offerta corrente?";
mMode = BootstrapModal.ModalMode.Confirm;
modalOpt = new();
modalOpt.Add(true, "Si");
modalOpt.Add(false, "No");
if (!await Modal!.ShowAsync<bool>())
return;
EditRecord = null;
SelRecord = null;
}
private async Task DoReset()
{
EditRecord = null;
SelRecord = null;
}
private void DoFilter()
{
if (searchValByState.Equals("All"))
{
ListFilt = AllRecords
.ToList();
}
else
{
ListFilt = AllRecords
.Where(x => x.OffertState.ToString().Equals(SearchVal))
.ToList();
}
//// verifico eventuali filtri
//ListFilt = AllRecords
// .Where(x => x.OffertState == OfferStates.Open || allState)
// .ToList();
if (!string.IsNullOrEmpty(searchVal))
{
ListFilt = ListFilt
.Where(x => x.Description.Contains(searchVal, StringComparison.InvariantCultureIgnoreCase) || x.OfferCode.Contains(searchVal, StringComparison.InvariantCultureIgnoreCase))
.ToList();
}
totalCount = ListFilt.Count();
currPage = 1;
}
/// <summary>
/// Salva record ed avanza compilazione
/// </summary>
/// <param name="updRec"></param>
/// <returns></returns>
private async Task DoSave(OfferModel updRec)
{
// salvo record
await OffService.UpsertAsync(updRec);
// cambio step
CompileStep nextStep = new CompileStep();
if (currStep < CompileStep.Delivery)
{
nextStep = currStep + 1;
AdvStep(nextStep);
}
else
{
EditRecord = null;
}
}
private void DoSelect(OfferModel curRec)
{
SelRecord = curRec;
}
private void EditState(OfferModel? curRec)
{
EditStateRec = curRec;
}
/// <summary>
/// Path da parent record
/// </summary>
/// <param name="objID"></param>
/// <returns></returns>
private string FolderPath(int objID)
{
return $"SO-{objID:X8}";
}
/// <summary>
/// Rilegge tabella
/// </summary>
private async Task ForceReload()
{
isLoading = true;
await Task.Delay(50);
await ReloadDataAsync();
await Task.Delay(50);
UpdateTable();
await Task.Delay(50);
isLoading = false;
await Task.Delay(50);
}
/// <summary>
/// Rilettura info di base
/// </summary>
/// <returns></returns>
private async Task ReloadBaseData()
{
AllConfEnvir = await EPService.GetAllAsync();
}
/// <summary>
/// Legge i dati dei record completi
/// </summary>
private async Task ReloadDataAsync()
{
await OffService.CheckExpiredAsync();
AllRecords = await OffService.GetFiltAsync(PeriodoSel.Inizio, PeriodoSel.Fine);
DoFilter();
}
private void SaveNumRec(int newNum)
{
numRecord = newNum;
UpdateTable();
}
private void SavePage(int newNum)
{
currPage = newNum;
UpdateTable();
}
private void SetDirty(EditStepDto updData)
{
#if false
// semodificato imposto colore variato x header
if (updData.Changed)
{
if (!ListtChanges.Contains(updData.SrcStep))
{
ListtChanges.Add(updData.SrcStep);
}
}
else
{
if (ListtChanges.Contains(updData.SrcStep))
{
ListtChanges.Remove(updData.SrcStep);
}
}
#endif
}
/// <summary>
/// Imposta periodo da filtro
/// </summary>
/// <param name="newPeriod"></param>
/// <returns></returns>
private async Task SetPeriodo(DtUtils.Periodo newPeriod)
{
PeriodoSel = newPeriod;
await ReloadDataAsync();
UpdateTable();
}
/// <summary>
/// Imposta lo stato dell'offerta VERIFICANDO i vari casi di stato di artenza/arrivo...
/// </summary>
/// <param name="currRec"></param>
/// <param name="newStatus"></param>
/// <returns></returns>
private async Task SetState(OfferModel currRec, OfferStates newStatus)
{
/* ---------------------------------
* se lo conferma: esegue step speciali come
* - generazione item prod con ID ed etichetta
* - generazione delle "buste di stima" x richiedere task stima preliminare
* - generazione della lista delle etichette + riga d'ordine da inviare
* - invio chiamata su channelRedis
*
* il puro invio dovrà poter essere fatto anche dalla tab ordini... e serve visualizzazione delle estim pending
* --------------------------------- */
// in primis: se è già confermata chiede una autorizzazione di conferma speciale
if (currRec.OffertState == OfferStates.Confirmed)
{
mTitle = "Conferma";
mMessage = "Confermi di voler modificare l'offerta già confermata?";
mMode = BootstrapModal.ModalMode.Confirm;
modalOpt = new();
modalOpt.Add(true, "Si");
modalOpt.Add(false, "No");
if (!await Modal!.ShowAsync<bool>())
return;
}
// se va verso conferma ricorda che ora l'ordine passa in pianificazione (carico macchine)
if (newStatus == OfferStates.Confirmed)
{
mTitle = "Conferma";
mMessage = $"Sei sicuro di voler confermare l'offerta e generare gli ordini? verranno generate anche le etichette per tutti i prodotti correlati ({currRec.NumProdItems})";
mMode = BootstrapModal.ModalMode.Confirm;
modalOpt = new();
modalOpt.Add(true, "Si");
modalOpt.Add(false, "No");
if (!await Modal!.ShowAsync<bool>())
return;
// creazione nuovo ordine da offerta
var newOrd = await OrdService.CloneOfferAsync(currRec);
if (newOrd != null)
{
// processo riga ordine x riga ordine creando per ogni riga una richiesta...
if (newOrd.OrderRowNav != null && newOrd.OrderRowNav.Count > 0)
{
// verifico parametri da conf envir...
var envRec = AllConfEnvir.FirstOrDefault(x => x.EnvirID == newOrd.Envir);
Egw.Window.Data.Enums.QuestionModes cMode = Egw.Window.Data.Enums.QuestionModes.ORDER;
Egw.Window.Data.Enums.QuestionOrderSubModes cSubMode = Egw.Window.Data.Enums.QuestionOrderSubModes.CREATE;
string reqKey = "";
foreach (var rigaOrd in newOrd.OrderRowNav)
{
// preparo le 2 richieste (creazione, stima)
CalcRequestDTO calcReq = new CalcRequestDTO();
bool needCalc = false;
// recupero elenco items collegati alla riga d'ordine
var ProdList = await PIService.GetByOrderRowAsync(rigaOrd.OrderRowID);
List<string> TagList = ProdList.Select(x => x.ProdItemTag).ToList();
//string serTagList = JsonConvert.SerializeObject(TagList);
string serTagList = string.Join(",", TagList);
// preparo richiesta serializzata e la accodo (viene inviata richiesta calcolo)
Dictionary<string, string> dictArgs = new Dictionary<string, string>();
// compongo righiesta
dictArgs.Add("UID", rigaOrd.OrderRowUID);
// creo registrazione richiesta...
var ruid = await CRService.AddRequestAsync($"{newOrd.Envir}", $"{(int)cMode}-{(int)cSubMode}", rigaOrd.OrderRowUID);
// aggiungo RUID effettivo
dictArgs.Add("RUID", ruid);
dictArgs.Add("OrderUID", rigaOrd.OrderNav.OrderCode);
dictArgs.Add("Mode", $"{(int)cMode}");
dictArgs.Add("TagList", serTagList);
// aggiungo file secondo ambiente...
string serKey = envRec != null ? envRec.SerStrucKey : "SerializedData";
switch (newOrd.Envir)
{
case EgwMultiEngineManager.Data.Constants.EXECENVIRONMENTS.WINDOW:
dictArgs.Add(serKey, rigaOrd.SerStruct);
needCalc = rigaOrd.SerStruct.Length > 2;
break;
case EgwMultiEngineManager.Data.Constants.EXECENVIRONMENTS.BEAM:
case EgwMultiEngineManager.Data.Constants.EXECENVIRONMENTS.WALL:
case EgwMultiEngineManager.Data.Constants.EXECENVIRONMENTS.CABINET:
// rileggo da file... 2check, spostare?!?
string folderPath = FolderPath(newOrd.OfferID);
string rawData = FileUtils.LoadFileContent(Path.Combine(basePath, folderPath), rigaOrd.FileResource);
dictArgs.Add(serKey, rawData);
dictArgs.Add("FileName", rigaOrd.FileName);
needCalc = rawData.Length > 2;
break;
case EgwMultiEngineManager.Data.Constants.EXECENVIRONMENTS.NULL:
default:
break;
}
// solo SE va calcolato...
if (needCalc)
{
// mando richiesta "create" preliminare con tutti i dati
dictArgs.Add("SubMode", $"{(int)cSubMode}");
calcReq = new CalcRequestDTO()
{
DictExec = dictArgs,
EnvType = newOrd.Envir
};
reqKey = $"{cMode}:{cSubMode}:{rigaOrd.OrderRowUID}";
await PService.EnqueueRequestAsync("Create", reqKey, calcReq);
// parto dalla history attuale
var currHist = newOrd.LogHistory;
// aggiungo evento...
currHist.Add(new TaskHistDTO()
{
DtEvent = DateTime.Now,
UserName = "Default User",
Message = $"{reqKey}",
IconCss = "fa-solid fa-hourglass-start"
});
newOrd.LogHistory = currHist;
//OrderHist = listOrdRow.LogHistory;
await OrdService.UpsertAsync(newOrd);
}
}
}
}
}
currRec.OffertState = newStatus;
await OffService.UpsertAsync(currRec);
await ReloadDataAsync();
UpdateTable();
}
private void SetupArrows()
{
listBord01 = new();
listBord01.Add("");
listBord01.Add("White");
listBord01.Add("");
listBord01.Add("");
}
/// <summary>
/// Filtro e paginazione
/// </summary>
private void UpdateTable()
{
// fix paginazione
ListRecords = ListFilt
.Skip(numRecord * (currPage - 1))
.Take(numRecord)
.ToList();
}
/// <summary>
/// Prepara URL x download report
/// </summary>
/// <param name="currId"></param>
/// <param name="repType"></param>
/// <param name="SelFile"></param>
/// <returns></returns>
private string DownloadUrl(int currId, string repType = "Offerta", string SelFile = "Offerta_01")
{
return $"{apiUrl}/download?ReqId={currId}&RepType={repType}&SelFile={SelFile}.repx";
}
private bool selectOptPrint = false;
private int currId = 0;
private void SelectOptPrint(int reqId)
{
selectOptPrint = true;
currId = reqId;
}
/// <summary>
/// Ritorno step Tree
/// </summary>
/// <param name="args"></param>
private void ClosePrintModal(bool args)
{
selectOptPrint = false;
}
private string ButtonStateCss(OfferModel item)
{
string ans = "";
switch (item.OffertState)
{
case OfferStates.Open:
ans = "info";
break;
case OfferStates.Confirmed:
ans = "success";
break;
case OfferStates.Expired:
ans = "warning";
break;
case OfferStates.Lost:
ans = "danger";
break;
}
return ans;
}
#endregion Private Methods
}
}