DeliveryConfirmation: enhance calculation
This commit is contained in:
@ -10,15 +10,21 @@ namespace Elwig.Documents {
|
|||||||
public int Year;
|
public int Year;
|
||||||
public IEnumerable<DeliveryPart> Deliveries;
|
public IEnumerable<DeliveryPart> Deliveries;
|
||||||
|
|
||||||
public DeliveryConfirmation(AppDbContext ctx, int year, Member m) : base($"Anlieferungsbestätigung {year} – {m.Name}", m) {
|
public DeliveryConfirmation(AppDbContext ctx, int year, Member m) :
|
||||||
|
base($"Anlieferungsbestätigung {year} – {((IAddress?)m.BillingAddress ?? m).Name}", m) {
|
||||||
Year = year;
|
Year = year;
|
||||||
DocumentId = $"Anl.-Best. {Year}/{m.MgNr}";
|
ShowDateAndLocation = true;
|
||||||
|
UseBillingAddress = true;
|
||||||
|
// FIXME footer in merged documents
|
||||||
|
//DocumentId = $"Anl.-Best. {Year}/{m.MgNr}";
|
||||||
Deliveries = ctx.DeliveryParts.FromSqlRaw($"""
|
Deliveries = ctx.DeliveryParts.FromSqlRaw($"""
|
||||||
SELECT p.*
|
SELECT p.*
|
||||||
FROM v_delivery v
|
FROM v_delivery v
|
||||||
JOIN delivery_part p ON (p.year, p.did, p.dpnr) = (v.year, v.did, v.dpnr)
|
JOIN delivery_part p ON (p.year, p.did, p.dpnr) = (v.year, v.did, v.dpnr)
|
||||||
WHERE (v.year, v.mgnr) = ({Year}, {m.MgNr})
|
WHERE (v.year, v.mgnr) = ({Year}, {m.MgNr})
|
||||||
ORDER BY v.sortid, v.abgewertet ASC, LENGTH(v.attributes) DESC, COALESCE(v.attributes, '~'), v.kmw DESC, v.lsnr, v.dpnr
|
ORDER BY v.sortid, v.abgewertet ASC,
|
||||||
|
COALESCE(LENGTH(v.attributes), 0) ASC, attribute_prio DESC, COALESCE(v.attributes, '~'),
|
||||||
|
v.kmw DESC, v.lsnr, v.dpnr
|
||||||
""")
|
""")
|
||||||
.ToList();
|
.ToList();
|
||||||
}
|
}
|
||||||
|
@ -46,6 +46,10 @@ table.delivery-confirmation tr.new td {
|
|||||||
border-top: 0.5pt solid black;
|
border-top: 0.5pt solid black;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
table.delivery-confirmation tr:not(.first) {
|
||||||
|
break-before: avoid;
|
||||||
|
}
|
||||||
|
|
||||||
table.delivery-confirmation tr:not(.first) td {
|
table.delivery-confirmation tr:not(.first) td {
|
||||||
padding-top: 0;
|
padding-top: 0;
|
||||||
}
|
}
|
||||||
|
@ -157,21 +157,25 @@ namespace Elwig.Helpers {
|
|||||||
ExecuteNonQuery(cnx, "DROP VIEW v_delivery");
|
ExecuteNonQuery(cnx, "DROP VIEW v_delivery");
|
||||||
ExecuteNonQuery(cnx, """
|
ExecuteNonQuery(cnx, """
|
||||||
CREATE VIEW v_delivery AS
|
CREATE VIEW v_delivery AS
|
||||||
SELECT p.year, p.did, p.dpnr,
|
SELECT s.*, GROUP_CONCAT(o.modid) AS modifiers
|
||||||
|
FROM (SELECT p.year, p.did, p.dpnr,
|
||||||
d.date, d.time, d.zwstid, d.lnr, d.lsnr,
|
d.date, d.time, d.zwstid, d.lnr, d.lsnr,
|
||||||
m.mgnr, m.family_name, m.given_name,
|
m.mgnr, m.family_name, m.given_name,
|
||||||
p.sortid, p.weight, p.kmw, ROUND(p.kmw * (4.54 + 0.022 * p.kmw), 0) AS oe, p.qualid, p.hkid, p.kgnr, p.rdnr,
|
p.sortid, p.weight, p.kmw, ROUND(p.kmw * (4.54 + 0.022 * p.kmw), 0) AS oe, p.qualid, p.hkid, p.kgnr, p.rdnr,
|
||||||
p.qualid IN (SELECT l.qualid FROM wine_quality_level l WHERE NOT l.predicate AND (p.kmw >= l.min_kmw OR l.min_kmw IS NULL) ORDER BY l.min_kmw DESC LIMIT 1,100) AS abgewertet,
|
p.qualid IN (SELECT l.qualid FROM wine_quality_level l WHERE NOT l.predicate AND (p.kmw >= l.min_kmw OR l.min_kmw IS NULL) ORDER BY l.min_kmw DESC LIMIT 1,100) AS abgewertet,
|
||||||
p.qualid NOT IN ('WEI', 'RSW', 'LDW') AS min_quw,
|
p.qualid NOT IN ('WEI', 'RSW', 'LDW') AS min_quw,
|
||||||
GROUP_CONCAT(DISTINCT a.attrid) as attributes, GROUP_CONCAT(DISTINCT o.modid) as modifiers,
|
GROUP_CONCAT(a.attrid) AS attributes,
|
||||||
d.comment, p.comment as part_comment
|
COALESCE(SUM(a.fill_lower_bins), 0) AS attribute_prio,
|
||||||
|
d.comment, p.comment AS part_comment
|
||||||
FROM delivery_part p
|
FROM delivery_part p
|
||||||
JOIN delivery d ON (d.year, d.did) = (p.year, p.did)
|
JOIN delivery d ON (d.year, d.did) = (p.year, p.did)
|
||||||
JOIN member m ON m.mgnr = d.mgnr
|
JOIN member m ON m.mgnr = d.mgnr
|
||||||
LEFT JOIN delivery_part_attribute a ON (a.year, a.did, a.dpnr) = (p.year, p.did, p.dpnr)
|
LEFT JOIN delivery_part_attribute pa ON (pa.year, pa.did, pa.dpnr) = (p.year, p.did, p.dpnr)
|
||||||
LEFT JOIN delivery_part_modifier o ON (o.year, o.did, o.dpnr) = (p.year, p.did, p.dpnr)
|
LEFT JOIN wine_attribute a ON a.attrid = pa.attrid
|
||||||
GROUP BY p.year, p.did, p.dpnr
|
GROUP BY p.year, p.did, p.dpnr) s
|
||||||
ORDER BY p.year, p.did, p.dpnr;
|
LEFT JOIN delivery_part_modifier o ON (o.year, o.did, o.dpnr) = (s.year, s.did, s.dpnr)
|
||||||
|
GROUP BY s.year, s.did, s.dpnr
|
||||||
|
ORDER BY s.year, s.did, s.dpnr;
|
||||||
""");
|
""");
|
||||||
|
|
||||||
ExecuteNonQuery(cnx, "DROP VIEW v_bucket");
|
ExecuteNonQuery(cnx, "DROP VIEW v_bucket");
|
||||||
@ -185,7 +189,9 @@ namespace Elwig.Helpers {
|
|||||||
ORDER BY year, mgnr, LENGTH(bin) DESC, bin;
|
ORDER BY year, mgnr, LENGTH(bin) DESC, bin;
|
||||||
""");
|
""");
|
||||||
|
|
||||||
ExecuteNonQuery(cnx, "ALTER TABLE wine_attribute ADD COLUMN fill_lower_bins INTEGER NOT NULL CHECK (fill_lower_bins IN (TRUE, FALSE)) DEFAULT FALSE");
|
ExecuteNonQuery(cnx, "ALTER TABLE wine_attribute ADD COLUMN fill_lower_bins INTEGER NOT NULL CHECK (fill_lower_bins IN (0, 1, 2)) DEFAULT 0");
|
||||||
|
|
||||||
|
ExecuteNonQuery(cnx, "UPDATE delivery_part_bin SET bin_2 = bin_2 + bin_3, bin_3 = NULL WHERE bin_4 IS NULL");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -43,7 +43,8 @@ namespace Elwig.Helpers.Billing {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public async Task CalculateBins() {
|
public async Task CalculateBins() {
|
||||||
var forcedAttr = Context.WineAttributes.Where(a => !a.FillLowerBins).Select(a => a.AttrId).ToArray();
|
var attrVals = Context.WineAttributes.ToDictionary(a => a.AttrId, a => a.FillLowerBins);
|
||||||
|
var attrForced = attrVals.Where(a => a.Value == 0).Select(a => a.Key).ToArray();
|
||||||
using var cnx = await AppDbContext.ConnectAsync();
|
using var cnx = await AppDbContext.ConnectAsync();
|
||||||
var memberOblRig = await GetMemberRightsObligations(cnx, Year);
|
var memberOblRig = await GetMemberRightsObligations(cnx, Year);
|
||||||
var inserts = new List<(int, int, int, int, int, int, int)>();
|
var inserts = new List<(int, int, int, int, int, int, int)>();
|
||||||
@ -54,7 +55,9 @@ namespace Elwig.Helpers.Billing {
|
|||||||
SELECT mgnr, did, dpnr, sortid, weight, kmw, qualid, attributes, modifiers
|
SELECT mgnr, did, dpnr, sortid, weight, kmw, qualid, attributes, modifiers
|
||||||
FROM v_delivery
|
FROM v_delivery
|
||||||
WHERE year = {Year}
|
WHERE year = {Year}
|
||||||
ORDER BY mgnr, sortid, abgewertet ASC, LENGTH(attributes) DESC, COALESCE(attributes, '~'), kmw DESC, lsnr, dpnr
|
ORDER BY mgnr, sortid, abgewertet ASC,
|
||||||
|
COALESCE(LENGTH(attributes), 0) ASC, attribute_prio DESC, COALESCE(attributes, '~'),
|
||||||
|
kmw DESC, lsnr, dpnr
|
||||||
""";
|
""";
|
||||||
var reader = await cmd.ExecuteReaderAsync();
|
var reader = await cmd.ExecuteReaderAsync();
|
||||||
while (await reader.ReadAsync()) {
|
while (await reader.ReadAsync()) {
|
||||||
@ -69,11 +72,18 @@ namespace Elwig.Helpers.Billing {
|
|||||||
|
|
||||||
int lastMgNr = 0;
|
int lastMgNr = 0;
|
||||||
Dictionary<string, int>? rights = null;
|
Dictionary<string, int>? rights = null;
|
||||||
|
Dictionary<string, int>? obligations = null;
|
||||||
|
Dictionary<string, int> used = new();
|
||||||
foreach (var (mgnr, did, dpnr, sortid, weight, kmw, qualid, attributes, modifiers) in deliveries) {
|
foreach (var (mgnr, did, dpnr, sortid, weight, kmw, qualid, attributes, modifiers) in deliveries) {
|
||||||
if (lastMgNr != mgnr) {
|
if (lastMgNr != mgnr) {
|
||||||
rights = memberOblRig.GetValueOrDefault(mgnr, (new(), new())).Item2;
|
var or = memberOblRig.GetValueOrDefault(mgnr, (new(), new()));
|
||||||
|
rights = or.Item2;
|
||||||
|
obligations = or.Item1;
|
||||||
|
used = new();
|
||||||
}
|
}
|
||||||
if (rights == null || rights.Count == 0 || qualid == "WEI" || qualid == "RSW" || qualid == "LDW") {
|
if (obligations == null || rights == null || obligations.Count == 0 || rights.Count == 0 ||
|
||||||
|
qualid == "WEI" || qualid == "RSW" || qualid == "LDW")
|
||||||
|
{
|
||||||
// Mitglied hat keine Flächenbindungen, oder
|
// Mitglied hat keine Flächenbindungen, oder
|
||||||
// Nicht mindestens Qualitätswein (QUW) -> ungebunden
|
// Nicht mindestens Qualitätswein (QUW) -> ungebunden
|
||||||
inserts.Add((did, dpnr, 0, 0, 0, 0, weight));
|
inserts.Add((did, dpnr, 0, 0, 0, 0, weight));
|
||||||
@ -85,21 +95,24 @@ namespace Elwig.Helpers.Billing {
|
|||||||
|
|
||||||
int w = weight;
|
int w = weight;
|
||||||
int[] b = new int[4];
|
int[] b = new int[4];
|
||||||
foreach (var p in Utils.Permutate(attributes, attributes.Intersect(forcedAttr))) {
|
foreach (var p in Utils.Permutate(attributes, attributes.Intersect(attrForced))) {
|
||||||
var c = p.Count();
|
var c = p.Count();
|
||||||
var key = sortid + string.Join("", p);
|
var key = sortid + string.Join("", p);
|
||||||
if (rights.ContainsKey(key)) {
|
if (rights.ContainsKey(key) && obligations.ContainsKey(key)) {
|
||||||
int i = 0;
|
int i = 0;
|
||||||
if (c == 1) {
|
if (c == 1) {
|
||||||
i = (p.ElementAt(0) == attributes[0]) ? 1 : 2;
|
i = (p.ElementAt(0) == attributes[0]) ? 1 : 2;
|
||||||
} else if (c == 0) {
|
} else if (c == 0) {
|
||||||
i = b.Length - 1;
|
i = b.Length - 1;
|
||||||
}
|
}
|
||||||
var v = Math.Max(0, Math.Min(rights[key], w));
|
var vr = Math.Max(0, Math.Min(rights[key] - used.GetValueOrDefault(key, 0), w));
|
||||||
|
var vo = Math.Max(0, Math.Min(obligations[key] - used.GetValueOrDefault(key, 0), w));
|
||||||
|
var v = (c == 0 || p.Select(a => attrVals[a]).Min() == 2) ? vr : vo;
|
||||||
b[i] += v;
|
b[i] += v;
|
||||||
rights[key] -= v;
|
used[key] = used.GetValueOrDefault(key, 0) + v;
|
||||||
w -= v;
|
w -= v;
|
||||||
}
|
}
|
||||||
|
if (w == 0) break;
|
||||||
}
|
}
|
||||||
inserts.Add((did, dpnr, b[0], b[1], b[2], b[3], weight - b[0] - b[1] - b[2] - b[3]));
|
inserts.Add((did, dpnr, b[0], b[1], b[2], b[3], weight - b[0] - b[1] - b[2] - b[3]));
|
||||||
lastMgNr = mgnr;
|
lastMgNr = mgnr;
|
||||||
|
@ -14,7 +14,7 @@ namespace Elwig.Models {
|
|||||||
public int? MaxKgPerHa { get; set; }
|
public int? MaxKgPerHa { get; set; }
|
||||||
|
|
||||||
[Column("fill_lower_bins")]
|
[Column("fill_lower_bins")]
|
||||||
public bool FillLowerBins { get; set; }
|
public int FillLowerBins { get; set; }
|
||||||
|
|
||||||
[Column("active")]
|
[Column("active")]
|
||||||
public bool IsActive { get; set; }
|
public bool IsActive { get; set; }
|
||||||
|
Reference in New Issue
Block a user