PaymentVariantsWindow: Add possibility to switch options on/off
This commit is contained in:
@ -9,7 +9,7 @@ namespace Elwig.Helpers {
|
||||
public static class AppDbUpdater {
|
||||
|
||||
// Don't forget to update value in Tests/fetch-resources.bat!
|
||||
public static readonly int RequiredSchemaVersion = 12;
|
||||
public static readonly int RequiredSchemaVersion = 13;
|
||||
|
||||
private static int VersionOffset = 0;
|
||||
|
||||
|
@ -23,12 +23,44 @@ namespace Elwig.Helpers.Billing {
|
||||
Schema = await JsonSchema.FromJsonAsync(stream ?? throw new ArgumentException("JSON schema not found"));
|
||||
}
|
||||
|
||||
public readonly JsonObject Data;
|
||||
|
||||
private readonly CalculationMode Mode;
|
||||
private readonly JsonObject Data;
|
||||
private readonly Dictionary<int, Curve> Curves;
|
||||
private readonly Dictionary<string, Curve> PaymentData;
|
||||
private readonly Dictionary<string, Curve> QualityData;
|
||||
|
||||
public bool ConsiderDelieryModifiers {
|
||||
get => GetConsider("consider_delivery_modifiers");
|
||||
set => SetConsider(value, "consider_delivery_modifiers");
|
||||
}
|
||||
public bool ConsiderContractPenalties {
|
||||
get => GetConsider("consider_contract_penalties");
|
||||
set => SetConsider(value, "consider_contract_penalties");
|
||||
}
|
||||
public bool ConsiderTotalPenalty {
|
||||
get => GetConsider("consider_total_penalty");
|
||||
set => SetConsider(value, "consider_total_penalty");
|
||||
}
|
||||
public bool ConsiderAutoBusinessShares {
|
||||
get => GetConsider("consider_auto_business_shares");
|
||||
set => SetConsider(value, "consider_total_penalty");
|
||||
}
|
||||
|
||||
private bool GetConsider(string name, string? wgMasterName = null) {
|
||||
return ((Mode == CalculationMode.Elwig) ? Data[name] : Data[wgMasterName ?? ""])?.AsValue().GetValue<bool>() ?? false;
|
||||
}
|
||||
|
||||
private void SetConsider(bool value, string name, string? wgMasterName = null) {
|
||||
if (Mode == CalculationMode.WgMaster && wgMasterName == null) {
|
||||
return;
|
||||
} else if (value) {
|
||||
Data[(Mode == CalculationMode.Elwig) ? name : wgMasterName ?? ""] = value;
|
||||
} else {
|
||||
Data.Remove((Mode == CalculationMode.Elwig) ? name : wgMasterName ?? "");
|
||||
}
|
||||
}
|
||||
|
||||
public BillingData(JsonObject data, IEnumerable<string> attributeVariants) {
|
||||
if (attributeVariants.Any(e => e.Any(c => c < 'A' || c > 'Z')))
|
||||
throw new ArgumentException("Invalid attributeVariants");
|
||||
@ -51,6 +83,10 @@ namespace Elwig.Helpers.Billing {
|
||||
}
|
||||
}
|
||||
|
||||
public static BillingData FromJson(string json) {
|
||||
return FromJson(json, []);
|
||||
}
|
||||
|
||||
public static BillingData FromJson(string json, IEnumerable<string> attributeVariants) {
|
||||
return new(ParseJson(json), attributeVariants);
|
||||
}
|
||||
|
@ -10,10 +10,19 @@ namespace Elwig.Helpers.Billing {
|
||||
|
||||
protected readonly int AvNr;
|
||||
protected readonly PaymentVar PaymentVariant;
|
||||
protected readonly BillingData Data;
|
||||
|
||||
public BillingVariant(int year, int avnr) : base(year) {
|
||||
AvNr = avnr;
|
||||
PaymentVariant = Context.PaymentVariants.Find(Year, AvNr) ?? throw new ArgumentException("PaymentVar not found");
|
||||
var attrVariants = Context.DeliveryParts
|
||||
.Where(d => d.Year == Year)
|
||||
.Select(d => $"{d.SortId}{d.AttrId}")
|
||||
.Distinct()
|
||||
.ToList()
|
||||
.Union(Context.WineVarieties.Select(v => v.SortId))
|
||||
.ToList();
|
||||
Data = BillingData.FromJson(PaymentVariant.Data, attrVariants);
|
||||
}
|
||||
|
||||
public async Task Calculate() {
|
||||
@ -22,7 +31,8 @@ namespace Elwig.Helpers.Billing {
|
||||
await DeleteInDb(cnx);
|
||||
await SetCalcTime(cnx);
|
||||
await CalculatePrices(cnx);
|
||||
await CalculateModifiers(cnx);
|
||||
if (Data.ConsiderDelieryModifiers)
|
||||
await CalculateDeliveryModifiers(cnx);
|
||||
await CalculateMemberModifiers(cnx);
|
||||
await tx.CommitAsync();
|
||||
}
|
||||
@ -39,7 +49,11 @@ namespace Elwig.Helpers.Billing {
|
||||
ROUND(p.amount / POW(10, s.precision - 2)) AS net_amount,
|
||||
ROUND(lp.amount / POW(10, s.precision - 2)) AS prev_amount,
|
||||
IIF(m.buchführend, s.vat_normal, s.vat_flatrate) AS vat,
|
||||
ROUND(COALESCE(u.total_penalty, 0) / POW(10, 4 - 2)) AS modifiers,
|
||||
ROUND(
|
||||
IIF({Data.ConsiderContractPenalties}, COALESCE(u.total_penalty, 0) / POW(10, 4 - 2), 0) +
|
||||
IIF({Data.ConsiderTotalPenalty}, COALESCE(b.total_penalty, 0), 0) +
|
||||
IIF({Data.ConsiderAutoBusinessShares}, 0, 0)
|
||||
) AS modifiers,
|
||||
lc.modifiers AS prev_modifiers
|
||||
FROM season s
|
||||
JOIN payment_variant v ON v.year = s.year
|
||||
@ -62,6 +76,14 @@ namespace Elwig.Helpers.Billing {
|
||||
FROM v_under_delivery u
|
||||
JOIN area_commitment_type t ON t.vtrgid = u.bucket
|
||||
GROUP BY year, mgnr) u ON (u.year, u.mgnr) = (s.year, m.mgnr)
|
||||
LEFT JOIN (SELECT s.year, u.mgnr,
|
||||
(COALESCE(IIF(u.weight = 0, -s.penalty_none, 0), 0) +
|
||||
COALESCE(IIF(u.diff < 0, -s.penalty_amount, 0), 0) +
|
||||
COALESCE(u.diff * s.penalty_per_kg, 0)
|
||||
) / POW(10, s.precision - 2) AS total_penalty
|
||||
FROM v_total_under_delivery u
|
||||
JOIN season s ON s.year = u.year
|
||||
WHERE u.diff < 0) b ON (b.year, b.mgnr) = (s.year, m.mgnr)
|
||||
WHERE s.year = {Year} AND v.avnr = {AvNr};
|
||||
|
||||
UPDATE payment_variant SET test_variant = FALSE WHERE (year, avnr) = ({Year}, {AvNr});
|
||||
@ -119,17 +141,6 @@ namespace Elwig.Helpers.Billing {
|
||||
}
|
||||
|
||||
protected async Task CalculatePrices(SqliteConnection cnx) {
|
||||
var jsonData = PaymentVariant.Data;
|
||||
var attrVariants = Context.DeliveryParts
|
||||
.Where(d => d.Year == Year)
|
||||
.Select(d => $"{d.SortId}{d.AttrId}")
|
||||
.Distinct()
|
||||
.ToList()
|
||||
.Union(Context.WineVarieties.Select(v => v.SortId))
|
||||
.ToList();
|
||||
|
||||
var data = BillingData.FromJson(jsonData, attrVariants);
|
||||
|
||||
var parts = new List<(int Year, int DId, int DPNr, int BktNr, string SortId, string Discr, int Value, double Oe, double Kmw, string QualId)>();
|
||||
using (var cmd = cnx.CreateCommand()) {
|
||||
cmd.CommandText = $"""
|
||||
@ -151,7 +162,7 @@ namespace Elwig.Helpers.Billing {
|
||||
var inserts = new List<(int Year, int DId, int DPNr, int BktNr, long Price, long Amount)>();
|
||||
foreach (var part in parts) {
|
||||
var attrId = (part.Discr == "_" || part.Discr == "") ? null : part.Discr;
|
||||
var price = data.CalculatePrice(part.SortId, attrId, part.QualId, part.Discr != "_", part.Oe, part.Kmw);
|
||||
var price = Data.CalculatePrice(part.SortId, attrId, part.QualId, part.Discr != "_", part.Oe, part.Kmw);
|
||||
var priceL = PaymentVariant.Season.DecToDb(price);
|
||||
inserts.Add((part.Year, part.DId, part.DPNr, part.BktNr, priceL, priceL * part.Value));
|
||||
}
|
||||
@ -162,7 +173,7 @@ namespace Elwig.Helpers.Billing {
|
||||
""");
|
||||
}
|
||||
|
||||
protected async Task CalculateModifiers(SqliteConnection cnx) {
|
||||
protected async Task CalculateDeliveryModifiers(SqliteConnection cnx) {
|
||||
await AppDbContext.ExecuteBatch(cnx, $"""
|
||||
INSERT INTO payment_delivery_part (year, did, dpnr, avnr, net_amount, mod_abs, mod_rel)
|
||||
SELECT d.year, d.did, d.dpnr, {AvNr}, 0, COALESCE(m.abs, 0), COALESCE(m.rel, 0)
|
||||
|
Reference in New Issue
Block a user