This commit is contained in:
wxt
2026-02-04 17:21:52 +08:00
4 changed files with 217 additions and 141 deletions

View File

@@ -1,23 +1,28 @@
using LiveCharts; using OxyPlot;
using LiveCharts.Wpf; using OxyPlot.Axes;
using OxyPlot.Series;
using OxyPlot.WindowsForms;
using System; using System;
using System.Collections.Generic;
using System.Drawing; using System.Drawing;
using System.Threading.Tasks; using System.Linq;
using System.Windows.Forms; using System.Windows.Forms;
using System.Windows.Forms.Integration;
namespace namespace
{ {
/// <summary> /// <summary>
/// 图表管理器 - 负责实时曲线图的创建和数据更新 /// 图表管理器 - 负责实时曲线图的创建和数据更新
/// 第一Y轴实时压力地址3130
/// 第二Y轴压力设定值地址2400对应"压力设置(PSI)"
/// </summary> /// </summary>
public class ChartManager public class ChartManager
{ {
private ElementHost _chartHost; private PlotView _plotView;
private LiveCharts.Wpf.CartesianChart _chart; private PlotModel _plotModel;
private ChartValues<double> _pressureValues; private LineSeries _pressureSeries; // 实时压力值地址3130
private ChartValues<double> _temperatureValues; private LineSeries _pressureSetSeries; // 压力设定值地址2400
private ChartValues<string> _timeLabels; private List<DataPoint> _pressureData;
private List<DataPoint> _pressureSetData;
private const int MAX_DATA_POINTS = 60; // 最多显示60个数据点 private const int MAX_DATA_POINTS = 60; // 最多显示60个数据点
/// <summary> /// <summary>
@@ -33,132 +38,167 @@ namespace 全自动水压检测仪
// 清除占位文字 // 清除占位文字
targetPanel.Text = null; targetPanel.Text = null;
// 初始化数据集合 // 初始化数据集合
_pressureValues = new ChartValues<double>(); _pressureData = new List<DataPoint>();
_temperatureValues = new ChartValues<double>(); _pressureSetData = new List<DataPoint>();
_timeLabels = new ChartValues<string>();
// 创建PlotModel
_plotModel = new PlotModel
{
Title = "",
Background = OxyColors.White,
PlotAreaBorderColor = OxyColors.Gray
};
// 配置图例
_plotModel.Legends.Add(new OxyPlot.Legends.Legend
{
LegendPosition = OxyPlot.Legends.LegendPosition.TopCenter,
LegendOrientation = OxyPlot.Legends.LegendOrientation.Horizontal
});
// 配置X轴时间轴 - 使用测试时间总秒数)
var xAxis = new LinearAxis
// 创建WPF图表控件 {
_chart = new LiveCharts.Wpf.CartesianChart Position = AxisPosition.Bottom,
Title = "测试时间",
TitleFontSize = 12,
FontSize = 10,
MajorGridlineStyle = LineStyle.Solid,
MajorGridlineColor = OxyColor.FromRgb(230, 230, 230),
MinorGridlineStyle = LineStyle.Dot,
MinorGridlineColor = OxyColor.FromRgb(240, 240, 240),
Minimum = 0,
Maximum = 60,
// 格式化X轴标签为HH:MM:SS格式
LabelFormatter = value =>
{ {
Background = System.Windows.Media.Brushes.White, int totalSec = (int)value;
DisableAnimations = false, int h = totalSec / 3600;
Hoverable = true, int m = (totalSec % 3600) / 60;
DataTooltip = null, int s = totalSec % 60;
LegendLocation = LegendLocation.Top return $"{h:D2}:{m:D2}:{s:D2}";
}; }
};
_plotModel.Axes.Add(xAxis);
// 配置左Y轴实时压力
var yAxisLeft = new LinearAxis
{
Position = AxisPosition.Left,
Title = "压力 (PSI)",
TitleFontSize = 12,
FontSize = 10,
TitleColor = OxyColors.Blue,
TextColor = OxyColors.Blue,
MajorGridlineStyle = LineStyle.Solid,
MajorGridlineColor = OxyColor.FromRgb(230, 230, 230),
MinorGridlineStyle = LineStyle.Dot,
MinorGridlineColor = OxyColor.FromRgb(240, 240, 240),
StringFormat = "F2",
Minimum = 0,
Key = "LeftAxis"
};
_plotModel.Axes.Add(yAxisLeft);
// 配置X轴时间轴 // 配置右Y轴压力设定值
_chart.AxisX.Add(new Axis var yAxisRight = new LinearAxis
{ {
Title = "时间", Position = AxisPosition.Right,
Labels = _timeLabels, Title = "压力设定值 (PSI)",
FontSize = 12, TitleFontSize = 12,
Foreground = System.Windows.Media.Brushes.Black, FontSize = 10,
Separator = new LiveCharts.Wpf.Separator TitleColor = OxyColors.Red,
{ TextColor = OxyColors.Red,
Step = 5, MajorGridlineStyle = LineStyle.None,
IsEnabled = true StringFormat = "F2",
} Minimum = 0,
}); Key = "RightAxis"
};
_plotModel.Axes.Add(yAxisRight);
// 配置Y轴压力 // 添加实时压力曲线
_chart.AxisY.Add(new Axis _pressureSeries = new LineSeries
{ {
Title = "压力 (MPa)", Title = "实时压力",
FontSize = 12, Color = OxyColors.Blue,
Foreground = System.Windows.Media.Brushes.Blue, StrokeThickness = 2,
LabelFormatter = value => value.ToString("F2"), MarkerType = MarkerType.Circle,
MinValue = 0 MarkerSize = 4,
}); MarkerFill = OxyColors.Blue,
YAxisKey = "LeftAxis"
};
_plotModel.Series.Add(_pressureSeries);
// 配置第二Y轴压力设定值 // 添加压力设定值曲线
_chart.AxisY.Add(new Axis _pressureSetSeries = new LineSeries
{ {
Title = "压力设定值 (MPa)", Title = "压力设定值",
FontSize = 12, Color = OxyColors.Red,
Foreground = System.Windows.Media.Brushes.Red, StrokeThickness = 2,
LabelFormatter = value => value.ToString("F2"), MarkerType = MarkerType.Diamond,
Position = AxisPosition.RightTop, MarkerSize = 4,
MinValue = 0 MarkerFill = OxyColors.Red,
}); YAxisKey = "RightAxis"
};
_plotModel.Series.Add(_pressureSetSeries);
// 添加压力曲线 // 创建PlotView控件
_chart.Series.Add(new LineSeries _plotView = new PlotView
{ {
Title = "实时压力", Model = _plotModel,
Values = _pressureValues, Dock = DockStyle.Fill,
Stroke = System.Windows.Media.Brushes.Blue, BackColor = System.Drawing.Color.White
Fill = System.Windows.Media.Brushes.Transparent, };
PointGeometry = DefaultGeometries.Circle,
PointGeometrySize = 5,
StrokeThickness = 2,
LineSmoothness = 0.3,
ScalesYAt = 0
});
// 添加压力设定值曲线 // 添加到目标面板
_chart.Series.Add(new LineSeries targetPanel.Controls.Add(_plotView);
{
Title = "压力设定值",
Values = _temperatureValues,
Stroke = System.Windows.Media.Brushes.Red,
Fill = System.Windows.Media.Brushes.Transparent,
PointGeometry = DefaultGeometries.Diamond,
PointGeometrySize = 5,
StrokeThickness = 2,
LineSmoothness = 0.3,
ScalesYAt = 1
});
// 创建ElementHost以承载WPF控件
_chartHost = new ElementHost
{
Dock = DockStyle.Fill,
Child = _chart
};
// 添加到目标面板
targetPanel.Controls.Add(_chartHost);
}));
} }
/// <summary> /// <summary>
/// 添加新的数据点 /// 添加新的数据点
/// </summary> /// </summary>
/// <param name="pressure">实时压力值</param> /// <param name="pressure">实时压力值</param>
/// <param name="temperature">压力设定值</param> /// <param name="pressureSetValue">压力设定值(来自"压力设置(PSI)"地址2400</param>
/// <param name="time">时间标签(可选,默认使用当前时间</param> /// <param name="totalSeconds">测试时间(总秒数</param>
public void AddDataPoint(double pressure, double temperature, string time = null) public void AddDataPoint(double pressure, double pressureSetValue, int totalSeconds)
{ {
if (_pressureValues == null || _temperatureValues == null || _timeLabels == null) if (_pressureSeries == null || _pressureSetSeries == null || _plotModel == null)
return; return;
// 使用当前时间作为默认标签 // 添加数据点使用测试时间总秒数作为X轴值
if (string.IsNullOrEmpty(time)) _pressureData.Add(new DataPoint(totalSeconds, pressure));
time = DateTime.Now.ToString("HH:mm:ss"); _pressureSetData.Add(new DataPoint(totalSeconds, pressureSetValue));
// 添加数据点
_pressureValues.Add(pressure);
_temperatureValues.Add(temperature);
_timeLabels.Add(time);
// 限制数据点数量,保持图表流畅 // 限制数据点数量,保持图表流畅
if (_pressureValues.Count > MAX_DATA_POINTS) if (_pressureData.Count > MAX_DATA_POINTS)
{ {
_pressureValues.RemoveAt(0); _pressureData.RemoveAt(0);
_temperatureValues.RemoveAt(0); _pressureSetData.RemoveAt(0);
_timeLabels.RemoveAt(0);
} }
// 更新系列数据
_pressureSeries.Points.Clear();
_pressureSeries.Points.AddRange(_pressureData);
_pressureSetSeries.Points.Clear();
_pressureSetSeries.Points.AddRange(_pressureSetData);
// 动态调整X轴范围
if (_pressureData.Count > 0)
{
var xAxis = _plotModel.Axes.FirstOrDefault(a => a.Position == AxisPosition.Bottom);
if (xAxis != null)
{
double minX = _pressureData.Min(p => p.X);
double maxX = _pressureData.Max(p => p.X);
xAxis.Minimum = minX;
xAxis.Maximum = maxX + 5; // 留一点余量
}
}
// 刷新图表
_plotModel.InvalidatePlot(true);
} }
/// <summary> /// <summary>
@@ -166,9 +206,28 @@ namespace 全自动水压检测仪
/// </summary> /// </summary>
public void ClearData() public void ClearData()
{ {
_pressureValues?.Clear(); _pressureData?.Clear();
_temperatureValues?.Clear(); _pressureSetData?.Clear();
_timeLabels?.Clear();
if (_pressureSeries != null)
{
_pressureSeries.Points.Clear();
}
if (_pressureSetSeries != null)
{
_pressureSetSeries.Points.Clear();
}
// 重置X轴范围
var xAxis = _plotModel?.Axes.FirstOrDefault(a => a.Position == AxisPosition.Bottom);
if (xAxis != null)
{
xAxis.Minimum = 0;
xAxis.Maximum = 60;
}
_plotModel?.InvalidatePlot(true);
} }
/// <summary> /// <summary>
@@ -177,21 +236,26 @@ namespace 全自动水压检测仪
/// <param name="isHighTemperatureMode">是否为高温模式</param> /// <param name="isHighTemperatureMode">是否为高温模式</param>
public void UpdateChartMode(bool isHighTemperatureMode) public void UpdateChartMode(bool isHighTemperatureMode)
{ {
if (_chart == null || _chart.Series.Count < 2) if (_plotModel == null || _plotModel.Series.Count < 2)
return; return;
// 压力设定值曲线标题保持不变 // 压力设定值曲线标题保持不变
var pressureSetSeries = _chart.Series[1] as LineSeries; if (_pressureSetSeries != null)
if (pressureSetSeries != null)
{ {
pressureSetSeries.Title = "压力设定值"; _pressureSetSeries.Title = "压力设定值";
} }
// 可以根据模式调整Y轴范围压力设定值范围 // 可以根据模式调整Y轴范围压力设定值范围
if (_chart.AxisY.Count > 1) foreach (var axis in _plotModel.Axes)
{ {
_chart.AxisY[1].MaxValue = 10; // 压力设定值通常不超过10MPa if (axis.Key == "RightAxis")
{
axis.Maximum = 10; // 压力设定值通常不超过10MPa
break;
}
} }
_plotModel.InvalidatePlot(true);
} }
/// <summary> /// <summary>
@@ -199,11 +263,12 @@ namespace 全自动水压检测仪
/// </summary> /// </summary>
public void Dispose() public void Dispose()
{ {
_chartHost?.Dispose(); _plotView?.Dispose();
_chart = null; _plotModel = null;
_pressureValues = null; _pressureSeries = null;
_temperatureValues = null; _pressureSetSeries = null;
_timeLabels = null; _pressureData = null;
_pressureSetData = null;
} }
} }
} }

View File

@@ -516,13 +516,13 @@ namespace 全自动水压检测仪
var pressure = c.UshortToFloat(modbusData.Real1[1], modbusData.Real1[0]); var pressure = c.UshortToFloat(modbusData.Real1[1], modbusData.Real1[0]);
var pressureSetValue = c.UshortToFloat(modbusData.Real12[1], modbusData.Real12[0]); var pressureSetValue = c.UshortToFloat(modbusData.Real12[1], modbusData.Real12[0]);
// 获取测试时间从PLC读取的时分秒 // 获取测试时间从PLC读取的时分秒并计算总秒数
int seconds = (modbusData.Real6 != null && modbusData.Real6.Length >= 1) ? modbusData.Real6[0] : 0; int seconds = (modbusData.Real6 != null && modbusData.Real6.Length >= 1) ? modbusData.Real6[0] : 0;
int minutes = (modbusData.Real7 != null && modbusData.Real7.Length >= 1) ? modbusData.Real7[0] : 0; int minutes = (modbusData.Real7 != null && modbusData.Real7.Length >= 1) ? modbusData.Real7[0] : 0;
int hours = (modbusData.Real8 != null && modbusData.Real8.Length >= 1) ? modbusData.Real8[0] : 0; int hours = (modbusData.Real8 != null && modbusData.Real8.Length >= 1) ? modbusData.Real8[0] : 0;
string testTime = $"{hours:D2}:{minutes:D2}:{seconds:D2}"; int totalSeconds = hours * 3600 + minutes * 60 + seconds;
_chartManager?.AddDataPoint(pressure, pressureSetValue, testTime); _chartManager?.AddDataPoint(pressure, pressureSetValue, totalSeconds);
} }
catch (Exception chartEx) catch (Exception chartEx)
{ {

View File

@@ -7,11 +7,14 @@
<package id="K4os.Compression.LZ4" version="1.3.8" targetFramework="net472" /> <package id="K4os.Compression.LZ4" version="1.3.8" targetFramework="net472" />
<package id="K4os.Compression.LZ4.Streams" version="1.3.8" targetFramework="net472" /> <package id="K4os.Compression.LZ4.Streams" version="1.3.8" targetFramework="net472" />
<package id="K4os.Hash.xxHash" version="1.0.8" targetFramework="net472" /> <package id="K4os.Hash.xxHash" version="1.0.8" targetFramework="net472" />
<package id="LiveCharts" version="0.9.7" targetFramework="net472" /> <package id="OxyPlot.Core" version="2.2.0" targetFramework="net472" />
<package id="LiveCharts.Wpf" version="0.9.7" targetFramework="net472" /> <package id="OxyPlot.WindowsForms" version="2.2.0" targetFramework="net472" />
<package id="Microsoft.Bcl.AsyncInterfaces" version="9.0.1" targetFramework="net472" /> <package id="Microsoft.Bcl.AsyncInterfaces" version="9.0.1" targetFramework="net472" />
<package id="Microsoft.NETFramework.ReferenceAssemblies" version="1.0.2" targetFramework="net472" developmentDependency="true" />
<package id="Microsoft.NETFramework.ReferenceAssemblies.net472" version="1.0.2" targetFramework="net472" developmentDependency="true" />
<package id="MySql.Data" version="9.5.0" targetFramework="net472" /> <package id="MySql.Data" version="9.5.0" targetFramework="net472" />
<package id="NModbus4" version="2.1.0" targetFramework="net472" /> <package id="NModbus4" version="2.1.0" targetFramework="net472" />
<package id="OxyPlot.Core" version="2.2.0" targetFramework="net472" />
<package id="SunnyUI" version="3.9.1" targetFramework="net472" /> <package id="SunnyUI" version="3.9.1" targetFramework="net472" />
<package id="SunnyUI.Common" version="3.9.1" targetFramework="net472" /> <package id="SunnyUI.Common" version="3.9.1" targetFramework="net472" />
<package id="System.Buffers" version="4.5.1" targetFramework="net472" /> <package id="System.Buffers" version="4.5.1" targetFramework="net472" />

View File

@@ -12,6 +12,8 @@
<FileAlignment>512</FileAlignment> <FileAlignment>512</FileAlignment>
<AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects> <AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects>
<Deterministic>true</Deterministic> <Deterministic>true</Deterministic>
<NuGetPackageImportStamp>
</NuGetPackageImportStamp>
</PropertyGroup> </PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' "> <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<PlatformTarget>AnyCPU</PlatformTarget> <PlatformTarget>AnyCPU</PlatformTarget>
@@ -59,11 +61,15 @@
<Reference Include="K4os.Hash.xxHash, Version=1.0.8.0, Culture=neutral, PublicKeyToken=32cd54395057cec3, processorArchitecture=MSIL"> <Reference Include="K4os.Hash.xxHash, Version=1.0.8.0, Culture=neutral, PublicKeyToken=32cd54395057cec3, processorArchitecture=MSIL">
<HintPath>..\packages\K4os.Hash.xxHash.1.0.8\lib\net462\K4os.Hash.xxHash.dll</HintPath> <HintPath>..\packages\K4os.Hash.xxHash.1.0.8\lib\net462\K4os.Hash.xxHash.dll</HintPath>
</Reference> </Reference>
<Reference Include="LiveCharts, Version=0.9.7.0, Culture=neutral, PublicKeyToken=0bc1f845d1ebb8df, processorArchitecture=MSIL"> <Reference Include="OxyPlot, Version=2.2.0.0, Culture=neutral, PublicKeyToken=638079a8f0bd61e9, processorArchitecture=MSIL">
<HintPath>..\packages\LiveCharts.0.9.7\lib\net45\LiveCharts.dll</HintPath> <HintPath>..\packages\OxyPlot.Core.2.2.0\lib\net462\OxyPlot.dll</HintPath>
<Private>True</Private>
<SpecificVersion>True</SpecificVersion>
</Reference> </Reference>
<Reference Include="LiveCharts.Wpf, Version=0.9.7.0, Culture=neutral, PublicKeyToken=0bc1f845d1ebb8df, processorArchitecture=MSIL"> <Reference Include="OxyPlot.WindowsForms, Version=2.2.0.0, Culture=neutral, PublicKeyToken=245eacd6b5d2d338, processorArchitecture=MSIL">
<HintPath>..\packages\LiveCharts.Wpf.0.9.7\lib\net45\LiveCharts.Wpf.dll</HintPath> <HintPath>..\packages\OxyPlot.WindowsForms.2.2.0\lib\net462\OxyPlot.WindowsForms.dll</HintPath>
<Private>True</Private>
<SpecificVersion>False</SpecificVersion>
</Reference> </Reference>
<Reference Include="Microsoft.Bcl.AsyncInterfaces, Version=9.0.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL"> <Reference Include="Microsoft.Bcl.AsyncInterfaces, Version=9.0.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.Bcl.AsyncInterfaces.9.0.1\lib\net462\Microsoft.Bcl.AsyncInterfaces.dll</HintPath> <HintPath>..\packages\Microsoft.Bcl.AsyncInterfaces.9.0.1\lib\net462\Microsoft.Bcl.AsyncInterfaces.dll</HintPath>
@@ -74,8 +80,6 @@
<Reference Include="NModbus4, Version=2.1.0.0, Culture=neutral, processorArchitecture=MSIL"> <Reference Include="NModbus4, Version=2.1.0.0, Culture=neutral, processorArchitecture=MSIL">
<HintPath>..\packages\NModbus4.2.1.0\lib\net40\NModbus4.dll</HintPath> <HintPath>..\packages\NModbus4.2.1.0\lib\net40\NModbus4.dll</HintPath>
</Reference> </Reference>
<Reference Include="PresentationCore" />
<Reference Include="PresentationFramework" />
<Reference Include="SunnyUI, Version=3.9.1.0, Culture=neutral, PublicKeyToken=3f4ebed4237db5e1, processorArchitecture=MSIL"> <Reference Include="SunnyUI, Version=3.9.1.0, Culture=neutral, PublicKeyToken=3f4ebed4237db5e1, processorArchitecture=MSIL">
<HintPath>..\packages\SunnyUI.3.9.1\lib\net472\SunnyUI.dll</HintPath> <HintPath>..\packages\SunnyUI.3.9.1\lib\net472\SunnyUI.dll</HintPath>
</Reference> </Reference>
@@ -121,9 +125,6 @@
<Reference Include="System.Net.Http" /> <Reference Include="System.Net.Http" />
<Reference Include="System.Windows.Forms" /> <Reference Include="System.Windows.Forms" />
<Reference Include="System.Xml" /> <Reference Include="System.Xml" />
<Reference Include="System.Xaml" />
<Reference Include="WindowsBase" />
<Reference Include="WindowsFormsIntegration" />
<Reference Include="ZstdSharp, Version=0.8.6.0, Culture=neutral, PublicKeyToken=8d151af33a4ad5cf, processorArchitecture=MSIL"> <Reference Include="ZstdSharp, Version=0.8.6.0, Culture=neutral, PublicKeyToken=8d151af33a4ad5cf, processorArchitecture=MSIL">
<HintPath>..\packages\ZstdSharp.Port.0.8.6\lib\net462\ZstdSharp.dll</HintPath> <HintPath>..\packages\ZstdSharp.Port.0.8.6\lib\net462\ZstdSharp.dll</HintPath>
</Reference> </Reference>
@@ -284,4 +285,11 @@
<WCFMetadata Include="Connected Services\" /> <WCFMetadata Include="Connected Services\" />
</ItemGroup> </ItemGroup>
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" /> <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
<Import Project="..\packages\Microsoft.NETFramework.ReferenceAssemblies.net472.1.0.2\build\Microsoft.NETFramework.ReferenceAssemblies.net472.targets" Condition="Exists('..\packages\Microsoft.NETFramework.ReferenceAssemblies.net472.1.0.2\build\Microsoft.NETFramework.ReferenceAssemblies.net472.targets')" />
<Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild">
<PropertyGroup>
<ErrorText>这台计算机上缺少此项目引用的 NuGet 程序包。使用“NuGet 程序包还原”可下载这些程序包。有关更多信息,请参见 http://go.microsoft.com/fwlink/?LinkID=322105。缺少的文件是 {0}。</ErrorText>
</PropertyGroup>
<Error Condition="!Exists('..\packages\Microsoft.NETFramework.ReferenceAssemblies.net472.1.0.2\build\Microsoft.NETFramework.ReferenceAssemblies.net472.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.NETFramework.ReferenceAssemblies.net472.1.0.2\build\Microsoft.NETFramework.ReferenceAssemblies.net472.targets'))" />
</Target>
</Project> </Project>