更新
This commit is contained in:
@@ -60,7 +60,7 @@ namespace TabletTester2025.Services
|
||||
{
|
||||
var data = batches.ToList();
|
||||
var sheet = package.Workbook.Worksheets.Add("硬度报表");
|
||||
WriteHeader(sheet, "检测时间", "样品名称", "平均值(N)", "平均偏差(N)", "RSD(%)", "最大值(N)", "最小值(N)", "测试次数", "单次数据(N)", "判定");
|
||||
WriteHeader(sheet, "检测时间", "样品名称", "平均值(N)", "平均偏差(N)", "RSD(%)", "最大值(N)", "测试次数", "单次数据(N)", "判定");
|
||||
|
||||
if (data.Count == 0)
|
||||
{
|
||||
@@ -78,10 +78,9 @@ namespace TabletTester2025.Services
|
||||
sheet.Cells[row, 4].Value = b.HardnessAverageDeviation;
|
||||
sheet.Cells[row, 5].Value = b.HardnessRSD;
|
||||
sheet.Cells[row, 6].Value = b.HardnessMax;
|
||||
sheet.Cells[row, 7].Value = b.HardnessMin;
|
||||
sheet.Cells[row, 8].Value = b.HardnessTestCount;
|
||||
sheet.Cells[row, 9].Value = b.HardnessSampleSummary;
|
||||
sheet.Cells[row, 10].Value = b.HardnessPassText;
|
||||
sheet.Cells[row, 7].Value = b.HardnessTestCount;
|
||||
sheet.Cells[row, 8].Value = b.HardnessSampleSummary;
|
||||
sheet.Cells[row, 9].Value = b.HardnessPassText;
|
||||
row++;
|
||||
}
|
||||
|
||||
|
||||
@@ -17,12 +17,13 @@ namespace TabletTester2025.Services
|
||||
// 模拟不同地址的数据
|
||||
float value = startAddress switch
|
||||
{
|
||||
72 => 40 + (float)_rand.NextDouble() * 20, // 硬度最大采集力
|
||||
100 => 40 + (float)_rand.NextDouble() * 20, // 硬度 40~60N
|
||||
410 => 100f, // 脆碎圈数
|
||||
412 => 5.0f + (float)_rand.NextDouble() * 2, // 脆碎度前重
|
||||
414 => 4.9f + (float)_rand.NextDouble() * 2, // 后重
|
||||
416 => 1.0f, // 失重率%
|
||||
300 => 37.0f, // 温度
|
||||
300 => 100.0f, // 硬度加压速度(mm/min)
|
||||
340 => 50f, // 溶出速度1(r/min)
|
||||
350 => 50f, // 溶出速度2(r/min)
|
||||
400 => 50 + (float)_rand.NextDouble() * 30, // 转速
|
||||
|
||||
@@ -64,7 +64,7 @@ namespace TabletTester2025.ViewModels
|
||||
Tester.ApplyPharmaDefaults();
|
||||
return Task.CompletedTask;
|
||||
});
|
||||
OpenHistoryCommand = new AsyncRelayCommand(() => { new HistoryWindow().ShowDialog(); return Task.CompletedTask; });
|
||||
OpenHistoryCommand = new AsyncRelayCommand(() => { new HistoryWindow(_db).ShowDialog(); return Task.CompletedTask; });
|
||||
}
|
||||
|
||||
private async Task ConnectToPlc()
|
||||
|
||||
@@ -1603,8 +1603,13 @@ namespace TabletTester2025.ViewModels
|
||||
throw new InvalidOperationException("硬度完成线圈未配置");
|
||||
if (ResolveHardnessLiveForceRegister() == 0)
|
||||
throw new InvalidOperationException("硬度实时力寄存器未配置");
|
||||
if (ResolveHardnessMaxRegister() == 0)
|
||||
throw new InvalidOperationException("硬度最大采集力寄存器未配置");
|
||||
if (ResolveHardnessSpeedRegister() == 0)
|
||||
throw new InvalidOperationException("硬度加压速度寄存器未配置");
|
||||
|
||||
await _plc.WriteFloatAsync(_plcConfig.HardnessSudu, (float)HardnessSudu);
|
||||
await LoadHardnessSpeedSettingAsync();
|
||||
await _plc.WriteFloatAsync(ResolveHardnessSpeedRegister(), (float)HardnessSudu);
|
||||
await _plc.WriteFloatAsync(_plcConfig.HardnessWeiyi, (float)HardnessWeiyi);
|
||||
|
||||
while (_isHardnessRunning && _hardnessResults.Count < count)
|
||||
@@ -1617,6 +1622,7 @@ namespace TabletTester2025.ViewModels
|
||||
|
||||
double value = await WaitForHardnessSamplePeakAsync(completeCoil);
|
||||
AddHardnessSample(value);
|
||||
await ReadHardnessMaxCaptureAsync();
|
||||
ApplyHardnessStatistics(count);
|
||||
await WaitForCoilStateAsync(completeCoil, false, TimeSpan.FromSeconds(10), "硬度完成信号未回落");
|
||||
}
|
||||
@@ -1624,6 +1630,7 @@ namespace TabletTester2025.ViewModels
|
||||
if (_hardnessResults.Count < count)
|
||||
throw new InvalidOperationException("硬度测试已停止,未保存结果");
|
||||
|
||||
await ReadHardnessMaxCaptureAsync();
|
||||
ApplyHardnessStatistics(count);
|
||||
AddHardnessGroupSummaryRow();
|
||||
resultReady = true;
|
||||
@@ -1663,6 +1670,20 @@ namespace TabletTester2025.ViewModels
|
||||
: (ushort)1314;
|
||||
}
|
||||
|
||||
private ushort ResolveHardnessMaxRegister()
|
||||
{
|
||||
return _plcConfig.HardnessMax != 0
|
||||
? _plcConfig.HardnessMax
|
||||
: (ushort)72;
|
||||
}
|
||||
|
||||
private ushort ResolveHardnessSpeedRegister()
|
||||
{
|
||||
return _plcConfig.HardnessSudu != 0
|
||||
? _plcConfig.HardnessSudu
|
||||
: (ushort)300;
|
||||
}
|
||||
|
||||
private async Task<double> ReadHardnessLiveForceAsync()
|
||||
{
|
||||
double value = await _plc.ReadFloatAsync(ResolveHardnessLiveForceRegister());
|
||||
@@ -1673,6 +1694,23 @@ namespace TabletTester2025.ViewModels
|
||||
return value;
|
||||
}
|
||||
|
||||
private async Task<double> ReadHardnessMaxCaptureAsync()
|
||||
{
|
||||
double value = await _plc.ReadFloatAsync(ResolveHardnessMaxRegister());
|
||||
if (!double.IsFinite(value) || value < 0)
|
||||
throw new InvalidOperationException("硬度最大采集力数据异常");
|
||||
|
||||
HardnessMax = value;
|
||||
return value;
|
||||
}
|
||||
|
||||
private async Task LoadHardnessSpeedSettingAsync()
|
||||
{
|
||||
double value = await _plc.ReadFloatAsync(ResolveHardnessSpeedRegister());
|
||||
if (double.IsFinite(value) && value > 0)
|
||||
HardnessSudu = value;
|
||||
}
|
||||
|
||||
private async Task<double> WaitForHardnessSamplePeakAsync(ushort completeCoil)
|
||||
{
|
||||
double peak = 0;
|
||||
@@ -1759,8 +1797,6 @@ namespace TabletTester2025.ViewModels
|
||||
HardnessAvg = stats.Average;
|
||||
HardnessAverageDeviation = stats.AverageDeviation;
|
||||
HardnessRSD = stats.RsdPercent;
|
||||
HardnessMax = stats.Maximum;
|
||||
HardnessMin = stats.Minimum;
|
||||
HardnessCurrentCount = stats.Count;
|
||||
HardnessPass = stats.IsPass;
|
||||
|
||||
@@ -2357,7 +2393,7 @@ namespace TabletTester2025.ViewModels
|
||||
TestType.Dissolution => string.IsNullOrWhiteSpace(effectiveDissolutionChannel) ? "溶出" : effectiveDissolutionChannel,
|
||||
_ => ""
|
||||
};
|
||||
LocalAlarm = $"{projectName}测试完成";
|
||||
LocalAlarm = $"{projectName}测试完成,已保存";
|
||||
});
|
||||
return true;
|
||||
}
|
||||
|
||||
@@ -199,7 +199,6 @@
|
||||
<DataGridTextColumn Header="平均偏差(N)" Binding="{Binding HardnessAverageDeviation, StringFormat=F2}" Width="110"/>
|
||||
<DataGridTextColumn Header="RSD(%)" Binding="{Binding HardnessRSD, StringFormat=F1}" Width="80"/>
|
||||
<DataGridTextColumn Header="最大值(N)" Binding="{Binding HardnessMax, StringFormat=F1}" Width="90"/>
|
||||
<DataGridTextColumn Header="最小值(N)" Binding="{Binding HardnessMin, StringFormat=F1}" Width="90"/>
|
||||
<DataGridTextColumn Header="测试次数" Binding="{Binding HardnessTestCount}" Width="70"/>
|
||||
<DataGridTextColumn Header="单次数据(N)" Binding="{Binding HardnessSampleSummary}" Width="220"/>
|
||||
<DataGridTextColumn Header="判定" Binding="{Binding HardnessPassText}" Width="70"/>
|
||||
|
||||
@@ -13,11 +13,10 @@ namespace TabletTester2025
|
||||
private readonly DatabaseService _dbService;
|
||||
private List<TestBatch> _allData;
|
||||
|
||||
public HistoryWindow()
|
||||
public HistoryWindow(DatabaseService dbService)
|
||||
{
|
||||
InitializeComponent();
|
||||
var connectionString = "Data Source=TabletTests.db";
|
||||
_dbService = new DatabaseService(connectionString);
|
||||
_dbService = dbService ?? throw new ArgumentNullException(nameof(dbService));
|
||||
LoadData();
|
||||
}
|
||||
|
||||
|
||||
@@ -235,22 +235,22 @@
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="">
|
||||
<Setter Property="Visibility" Value="Collapsed"/>
|
||||
</DataTrigger>
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="硬度测试完成">
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="硬度测试完成,已保存">
|
||||
<Setter Property="Foreground" Value="#1565C0"/>
|
||||
</DataTrigger>
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="脆碎度测试完成">
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="脆碎度测试完成,已保存">
|
||||
<Setter Property="Foreground" Value="#1565C0"/>
|
||||
</DataTrigger>
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="崩解测试完成">
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="崩解测试完成,已保存">
|
||||
<Setter Property="Foreground" Value="#1565C0"/>
|
||||
</DataTrigger>
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="溶出测试完成">
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="溶出测试完成,已保存">
|
||||
<Setter Property="Foreground" Value="#1565C0"/>
|
||||
</DataTrigger>
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="溶出1测试完成">
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="溶出1测试完成,已保存">
|
||||
<Setter Property="Foreground" Value="#1565C0"/>
|
||||
</DataTrigger>
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="溶出2测试完成">
|
||||
<DataTrigger Binding="{Binding LocalAlarm}" Value="溶出2测试完成,已保存">
|
||||
<Setter Property="Foreground" Value="#1565C0"/>
|
||||
</DataTrigger>
|
||||
</Style.Triggers>
|
||||
@@ -260,6 +260,8 @@
|
||||
|
||||
<TabControl Grid.Row="1" FontSize="13" BorderThickness="0">
|
||||
<TabItem Header="硬度">
|
||||
<ScrollViewer VerticalScrollBarVisibility="Auto"
|
||||
HorizontalScrollBarVisibility="Disabled">
|
||||
<Grid Margin="4,14,4,4">
|
||||
<Grid.RowDefinitions>
|
||||
<RowDefinition Height="Auto"/>
|
||||
@@ -304,12 +306,6 @@
|
||||
<TextBlock Text="{Binding HardnessMax, StringFormat=F1}" Foreground="#1565C0" Style="{StaticResource MetricValue}"/>
|
||||
</StackPanel>
|
||||
</Border>
|
||||
<Border Style="{StaticResource MetricCard}">
|
||||
<StackPanel>
|
||||
<TextBlock Text="最小力值(N)" Style="{StaticResource MetricLabel}"/>
|
||||
<TextBlock Text="{Binding HardnessMin, StringFormat=F1}" Foreground="#2E7D32" Style="{StaticResource MetricValue}"/>
|
||||
</StackPanel>
|
||||
</Border>
|
||||
<Border Style="{StaticResource MetricCard}">
|
||||
<StackPanel>
|
||||
<TextBlock Text="平均值(N)" Style="{StaticResource MetricLabel}"/>
|
||||
@@ -465,6 +461,7 @@
|
||||
<Button Command="{Binding ClearHardnessRecordsCommand}" Content="清空记录" Style="{StaticResource SecondaryButton}"/>
|
||||
</WrapPanel>
|
||||
</Grid>
|
||||
</ScrollViewer>
|
||||
</TabItem>
|
||||
|
||||
<TabItem Header="脆碎度">
|
||||
|
||||
@@ -83,6 +83,10 @@
|
||||
<TextBlock Text="加压压力:" Style="{StaticResource ParamLabel}"/>
|
||||
<TextBox x:Name="HardnessPressureBox"/>
|
||||
</StackPanel>
|
||||
<StackPanel Style="{StaticResource ParamRow}">
|
||||
<TextBlock Text="加压速度(mm/min):" Style="{StaticResource ParamLabel}"/>
|
||||
<TextBox x:Name="HardnessSpeedBox"/>
|
||||
</StackPanel>
|
||||
<StackPanel Style="{StaticResource ParamRow}">
|
||||
<TextBlock Text="硬度破损判定(N):" Style="{StaticResource ParamLabel}"/>
|
||||
<TextBox x:Name="HardnessDamageThresholdBox" helpers:NumericInput.AllowDecimal="True"/>
|
||||
|
||||
@@ -19,6 +19,7 @@ namespace TabletTester2025
|
||||
private async void SettingsWindow_Loaded(object sender, RoutedEventArgs e)
|
||||
{
|
||||
await LoadHardnessPressureAsync();
|
||||
await LoadHardnessSpeedAsync();
|
||||
await LoadHardnessDamageThresholdAsync();
|
||||
|
||||
// 脆碎度
|
||||
@@ -69,6 +70,7 @@ namespace TabletTester2025
|
||||
};
|
||||
p.HardnessTestCount = int.Parse(HardnessCountBox.Text);
|
||||
double hardnessPressure = ParseFiniteDouble(HardnessPressureBox.Text, "加压压力");
|
||||
double hardnessSpeed = ParsePositiveDouble(HardnessSpeedBox.Text, "加压速度");
|
||||
double hardnessDamageThreshold = ParsePositiveDouble(HardnessDamageThresholdBox.Text, "硬度破损判定");
|
||||
double friabilityRpm = ParseFiniteDouble(FriabilityRpmBox.Text, "脆碎度转速");
|
||||
p.FriabilityTargetRpm = friabilityRpm;
|
||||
@@ -86,6 +88,7 @@ namespace TabletTester2025
|
||||
|
||||
ValidateParameters(p);
|
||||
await WriteHardnessPressureAsync(hardnessPressure);
|
||||
await WriteHardnessSpeedAsync(hardnessSpeed);
|
||||
await WriteHardnessDamageThresholdAsync(hardnessDamageThreshold);
|
||||
await WriteFriabilityRpmAsync(friabilityRpm);
|
||||
await WriteDisintegrationTimeAsync(disintegrationTimeMin);
|
||||
@@ -202,6 +205,38 @@ namespace TabletTester2025
|
||||
return App.PlcConfig.HardnessPressure != 0 ? App.PlcConfig.HardnessPressure : (ushort)1480;
|
||||
}
|
||||
|
||||
private async Task LoadHardnessSpeedAsync()
|
||||
{
|
||||
ushort registerAddress = ResolveHardnessSpeedRegister();
|
||||
if (registerAddress == 0)
|
||||
return;
|
||||
|
||||
try
|
||||
{
|
||||
float value = await App.PlcService.ReadFloatAsync(registerAddress);
|
||||
if (float.IsFinite(value) && value > 0)
|
||||
HardnessSpeedBox.Text = value.ToString("0.###");
|
||||
}
|
||||
catch
|
||||
{
|
||||
HardnessSpeedBox.Text = "";
|
||||
}
|
||||
}
|
||||
|
||||
private static async Task WriteHardnessSpeedAsync(double value)
|
||||
{
|
||||
ushort registerAddress = ResolveHardnessSpeedRegister();
|
||||
if (registerAddress == 0)
|
||||
throw new InvalidOperationException("加压速度PLC寄存器地址未配置。");
|
||||
|
||||
await App.PlcService.WriteFloatAsync(registerAddress, (float)value);
|
||||
}
|
||||
|
||||
private static ushort ResolveHardnessSpeedRegister()
|
||||
{
|
||||
return App.PlcConfig.HardnessSudu != 0 ? App.PlcConfig.HardnessSudu : (ushort)300;
|
||||
}
|
||||
|
||||
private async Task LoadHardnessDamageThresholdAsync()
|
||||
{
|
||||
ushort registerAddress = ResolveHardnessDamageThresholdRegister();
|
||||
|
||||
Reference in New Issue
Block a user