feat: единый файл Лист1+СПРАВКА, фильтры, сортировка, живой п/п

- Загрузка одного .xlsx (СПРАВКА + Лист1 в одном файле)
- Рег. номера берутся из СПРАВКА автоматически, группируются по (ТН ВЭД, Страна)
- Отдельный опциональный справочник кодов (CodesImportService)
- Фильтры: Все / Авто / Проверить / Нет кода / Без нетто
- Таблица сразу отсортирована по ТН ВЭД с живым обновлением п/п
- Начальный п/п: поле в UI, автоинкремент после экспорта
- Лист2: рамки, жёлтая строка при >1 рег. номере, итого по кол-ву
- Лист3: ТН ВЭД в колонке B вместо константы 09035
- Красный ErrorMessage под кнопкой загрузки, удалён SpravkaFileEntry
This commit is contained in:
Dianaka123
2026-04-07 11:47:31 +03:00
parent addf55e3b2
commit 697ae44519
13 changed files with 602 additions and 429 deletions
@@ -15,9 +15,15 @@ public class CodeLookupService
_entries = _persistence.Load();
}
public void Reload() => _entries = _persistence.Load();
// Загружает коды из внешнего xlsx-справочника (заменяет текущие записи)
public void LoadFromEntries(List<CodeLookupEntry> entries)
{
_entries = entries;
}
// Назначает декларационные коды всем позициям в списке.
public bool HasEntries => _entries.Count > 0;
// Назначает декларационные коды всем позициям в списке
public void AssignCodes(IEnumerable<DeclarationItem> items)
{
foreach (var item in items)
@@ -46,18 +52,8 @@ public class CodeLookupService
// Несколько кодов — неоднозначность
item.CandidateCodes = new List<string>(entry.Codes);
// Эвристика: если у позиции есть рег. номер — предпочесть 9000
if (!string.IsNullOrEmpty(item.RegNumber) && entry.Codes.Contains("9000"))
{
item.DeclarationCode = "9000";
item.Confidence = ConfidenceLevel.Auto;
}
else
{
item.DeclarationCode = entry.Codes[0]; // первый = наиболее вероятный
item.Confidence = ConfidenceLevel.Review;
}
item.DeclarationCode = entry.Codes[0];
item.Confidence = ConfidenceLevel.Review;
}
// Фиксирует ручной выбор кода декларантом