Files
HC_APTBS/App.xaml.cs
LucianoDev 827b811b39 feat: developer tools page, auto-test orchestrator, BIP display, tests redesign
Bundles several feature streams that have been iterating on the working tree:

- Developer Tools page (Debug-only via DEVELOPER_TOOLS symbol): hosts the
  identification card, manual KWP write + transaction log, ROM/EEPROM dump
  card with progress banner and completion message, persisted custom-commands
  library, persisted EEPROM passwords library. New service primitives:
  IKwpService.SendRawCustomAsync / ReadEepromAsync / ReadRomEepromAsync.
  Persistence mirrors the Clients XML pattern in two new files
  (custom_commands.xml, eeprom_passwords.xml).
- Auto-test orchestrator (IAutoTestOrchestrator + AutoTestState): linear
  K-Line read -> unlock -> bench-on -> test sequence with snackbar UI and
  progress dialog VM, gated on dashboard alarms.
- BIP-STATUS display: BipDisplayViewModel + BipDisplayView, RAM read at
  0x0106 via IKwpService.ReadBipStatusAsync; status definitions in
  BipStatusDefinition.
- Tests page redesign: TestSectionCard + PhaseTileView replacing the old
  TestPlanView/TestRunningView/TestDoneView/TestPreconditionsView/
  TestSectionView controls and their VMs.
- Pump command sliders: Fluent thick-track style with overhang thumb,
  click-anywhere-and-drag, mouse-wheel adjustment.
- Window startup: app.manifest declares PerMonitorV2 DPI awareness,
  MainWindow installs a WM_GETMINMAXINFO hook in OnSourceInitialized and
  maximizes there (after the hook is in place) so the app fits the work
  area exactly on any display configuration.
- Misc: PercentToPixelsConverter, seed_aliases.py one-shot pump-alias
  importer, tools/Import-BipStatus.ps1, kline_eeprom_spec.md and
  dump-functions reference docs.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-05-07 13:59:50 +02:00

102 lines
4.7 KiB
C#

using System.Windows;
using System.Windows.Media;
using HC_APTBS.Infrastructure.Logging;
using Wpf.Ui.Appearance;
using HC_APTBS.Infrastructure.Pcan;
using HC_APTBS.Models;
using HC_APTBS.Services;
using HC_APTBS.Services.Impl;
using HC_APTBS.ViewModels;
using Microsoft.Extensions.DependencyInjection;
using Peak.Can.Basic;
namespace HC_APTBS;
/// <summary>
/// Application entry-point and DI container host.
///
/// <para>
/// Registers all services, ViewModels, and the main window with
/// <see cref="ServiceCollection"/>, then resolves and shows <see cref="MainWindow"/>
/// on startup. The container is disposed when the application exits.
/// </para>
/// </summary>
public partial class App : Application
{
private ServiceProvider? _serviceProvider;
protected override async void OnStartup(StartupEventArgs e)
{
base.OnStartup(e);
// Apply WPF-UI theme and accent colour before any UI is constructed.
ApplicationThemeManager.Apply(ApplicationTheme.Light);
ApplicationAccentColorManager.Apply(Color.FromRgb(0x21, 0x96, 0xF3));
var services = new ServiceCollection();
ConfigureServices(services);
_serviceProvider = services.BuildServiceProvider();
// Load the persisted language dictionary before any VM or view reads a string.
_ = _serviceProvider.GetRequiredService<ILocalizationService>();
// Wire runtime-warning hooks on pure Model classes before any config is loaded.
// Keeps the Models layer DI-free while still routing warnings through IAppLogger.
var logger = _serviceProvider.GetRequiredService<IAppLogger>();
CanBusParameter.WarningLogger = logger.Warning;
SensorConfiguration.WarningLogger = logger.Warning;
// Initialise the ViewModel (loads pump IDs, starts refresh timer, connects CAN).
var mainVm = _serviceProvider.GetRequiredService<MainViewModel>();
await mainVm.InitialiseAsync();
var window = _serviceProvider.GetRequiredService<MainWindow>();
window.Show();
}
protected override void OnExit(ExitEventArgs e)
{
_serviceProvider?.Dispose();
base.OnExit(e);
}
// ── Service registration ──────────────────────────────────────────────────
private static void ConfigureServices(IServiceCollection services)
{
// ── Infrastructure ────────────────────────────────────────────────────
services.AddSingleton<IAppLogger, AppLogger>();
// PcanAdapter requires a channel handle and baudrate read from configuration.
services.AddSingleton<ICanService>(sp =>
{
var cfg = sp.GetRequiredService<IConfigurationService>();
var logger = sp.GetRequiredService<IAppLogger>();
var channel = cfg.Bench.Channel;
// Default to 500 kbps; the channel can switch at runtime via ICanService.SwitchBaudrate.
return new PcanAdapter(channel, TPCANBaudrate.PCAN_BAUD_500K, logger);
});
services.AddSingleton<IKwpService, KwpService>();
// ── Application services ──────────────────────────────────────────────
services.AddSingleton<IConfigurationService, ConfigurationService>();
services.AddSingleton<IBenchService, BenchService>();
services.AddSingleton<IUnlockService, UnlockService>();
services.AddSingleton<ICalibrationService, CalibrationService>();
services.AddSingleton<IPdfService, PdfService>();
services.AddSingleton<ILocalizationService, LocalizationService>();
// Auto-test orchestrator. Factory indirection breaks the Orchestrator↔MainViewModel
// construction cycle (orchestrator needs IAutoTestHost, MainViewModel implements it).
services.AddSingleton<System.Func<IAutoTestHost>>(sp => () => sp.GetRequiredService<MainViewModel>());
services.AddSingleton<IAutoTestOrchestrator, AutoTestOrchestrator>();
// ── ViewModels ────────────────────────────────────────────────────────
services.AddSingleton<MainViewModel>();
// ── Views ─────────────────────────────────────────────────────────────
services.AddSingleton<MainWindow>();
}
}