using DevExpress.Utils; using DevExpress.Utils.Drawing; using DevExpress.XtraCharts; namespace Yw.WinFrmUI.Phart { /// /// 泵变速视图图表 /// public partial class PumpVariableSpeedViewChart : DevExpress.XtraEditors.XtraUserControl { public PumpVariableSpeedViewChart() { InitializeComponent(); InitialChart(); this.chartControl1.RuntimeHitTesting = true; } #region Private Variable private readonly string _tag_qh = "QH", _tag_qe = "QE", _tag_qp = "QP"; private XYDiagram _diagram; private XYDiagramDefaultPane _default_pane; private XYDiagramPane _bottom_pane; private AxisX _axis_x_flow; private AxisY _axis_y_head; private SecondaryAxisY _axis_y_eff, _axis_y_power; private PumpCoordinate _coordinate; private bool _qe_visible = true; private bool _qp_visible = true; private bool _equip_visible = true; private Yw.Geometry.CubicSpline2d _equip_line = null; private Yw.Geometry.Point2d _equip_pt = null; private List> _equip_sect_pt_list; private bool _initial_data = false; private PumpVariableSpeedViewViewModel _vm = null; #endregion #region Public Evnet /// /// 坐标变换事件 /// public event Action CoordinateChangedEvent = null; #endregion #region Initial /// /// 初始化图表 /// private void InitialChart() { this.chartControl1.SetChartDisplay(); this.chartControl1.Legend.Direction = DevExpress.XtraCharts.LegendDirection.TopToBottom; _diagram = (XYDiagram)this.chartControl1.Diagram; _default_pane = _diagram.DefaultPane; _bottom_pane = (XYDiagramPane)_diagram.FindPaneByName("BottomPanel"); _axis_x_flow = _diagram.AxisX; _axis_x_flow.SetAxisXQDisplay(); _axis_y_head = _diagram.AxisY; _axis_y_head.SetAxisYQHDisplay(); _axis_y_eff = _diagram.SecondaryAxesY.GetAxisByName("AxisYQE"); _axis_y_eff.SetSecondaryAxisYQEDisplay(); _axis_y_eff.Alignment = AxisAlignment.Far; _axis_y_power = _diagram.SecondaryAxesY.GetAxisByName("AxisYQP"); _axis_y_power.SetSecondaryAxisYQPDisplay(); this.chartControl1.SetChartMonoColorDisplay(); _axis_x_flow.Visibility = DefaultBoolean.False; _axis_x_flow.GridLines.Visible = false; _axis_y_head.Visibility = DefaultBoolean.False; _axis_y_head.GridLines.Visible = false; _axis_y_eff.Visibility = DefaultBoolean.False; _axis_y_eff.GridLines.Visible = false; _axis_y_power.Visibility = DefaultBoolean.False; _axis_y_power.GridLines.Visible = false; this.chartControl1.CustomPaint += ChartControl1_CustomPaint; } private void ChartControl1_CustomPaint(object sender, CustomPaintEventArgs e) { if (e is not DXCustomPaintEventArgs dxArgs) return; if (_vm == null) return; if (_equip_line != null && _equip_sect_pt_list != null && _equip_sect_pt_list.Any() && _equip_visible) { using Pen pen = new(Color.Black, 2); pen.DashStyle = System.Drawing.Drawing2D.DashStyle.DashDotDot; foreach (var tuple in _equip_sect_pt_list) { var color = tuple.Item2; var flow = tuple.Item3; var head = tuple.Item4; var eff = tuple.Item5; var power = tuple.Item6; DrawWorkPoint(dxArgs.Cache, color, _axis_y_head, flow, head); if (eff.HasValue && _qe_visible) DrawWorkPoint(dxArgs.Cache, color, _axis_y_eff, flow, eff.Value); if (power.HasValue && _qp_visible) DrawWorkPoint(dxArgs.Cache, color, _axis_y_power, flow, power.Value); } DrawEquipLine(dxArgs.Cache, pen, _equip_line); } } private void DrawEquipLine(GraphicsCache cache, Pen pen, Yw.Geometry.CubicSpline2d qh) { if (qh == null) return; var pt_list = qh.GetPointList().ToArray(); var pt_f_list = new List(); foreach (var pt in pt_list) { var x = pt.X; var y = pt.Y; var c_pt = _diagram.DiagramToPoint(x, y, _axis_x_flow, _axis_y_head); pt_f_list.Add(new PointF(c_pt.Point.X, c_pt.Point.Y)); } using var path = new System.Drawing.Drawing2D.GraphicsPath(); path.AddCurve(pt_f_list.ToArray()); cache.DrawPath(pen, path); } private void DrawWorkPoint(GraphicsCache cache, Color color, AxisYBase axis_y, double x, double y) { var dg_pt = _diagram.DiagramToPoint(x, y, _axis_x_flow, axis_y); cache.FillEllipse((int)(dg_pt.Point.X - 5), (int)(dg_pt.Point.Y - 5), 10, 10, color); } /// /// 初始化图表数据 /// public void InitialChartData() { _initial_data = false; _coordinate = null; UpdateChart(false); } #endregion #region Set /// /// 绑定数据 /// public void SetBindingData(PumpVariableSpeedViewViewModel vm) { _vm = vm; _initial_data = vm != null; UpdateChart(true); } /// /// 设置设计点 /// public void SetDesignPoint(double x, double y, double? start_head = null) { _equip_pt = new Geometry.Point2d(x, y); _equip_line = null; _equip_sect_pt_list = null; if (_vm == null) { return; } start_head ??= _coordinate.DispMinH(); var equip_line = Yw.Geometry.EquipCurveHelper.CalcEquipCurve(_vm.CurveQH, _equip_pt, start_head.Value, out Yw.Geometry.Point2d sect_pt); if (equip_line == null || sect_pt == null) return; _equip_line = equip_line; _equip_sect_pt_list = new List>(); double sect_flow = sect_pt.X; double sect_head = _vm.CurveQH.GetPointYUnlimited(sect_flow); double? sect_eff = null, sect_power = null; if (_vm.CurveQP != null) { sect_power = _vm.CurveQP.GetPointYUnlimited(sect_flow); //sect_eff = PumpCalcHelper.CalculateE(sect_flow, sect_head, sect_power.Value); } if (_vm.CurveQE != null) { sect_eff = _vm.CurveQE.GetPointYUnlimited(sect_flow); // sect_power = PumpCalcHelper.CalculateP(sect_flow, sect_head, sect_eff.Value); } _equip_sect_pt_list.Add(new(_vm.Id, _vm.Color, sect_flow, sect_head, sect_eff, sect_power)); if (_vm.Items != null && _vm.Items.Any()) { foreach (var vm in _vm.Items) { double flow = 0, head = 0; double? eff = null, power = null; var pump_equip_line = Yw.Geometry.EquipCurveHelper.CalcEquipCurve(vm.CurveQH, _equip_pt, start_head.Value, out Yw.Geometry.Point2d pump_sect_pt); if (pump_equip_line == null || pump_sect_pt == null) return; flow = pump_sect_pt.X; head = vm.CurveQH.GetPointYUnlimited(flow); if (vm.CurveQP != null) { power = vm.CurveQP.GetPointYUnlimited(flow); //eff = PumpCalcHelper.CalculateE(flow, head, power.Value); } if (vm.CurveQE != null) { eff = vm.CurveQE.GetPointYUnlimited(flow); //power = PumpCalcHelper.CalculateP(flow, head, eff.Value); } _equip_sect_pt_list.Add(new(vm.Id, vm.Color, flow, head, eff, power)); } } UpdateChart(true); } /// /// 清空曲线 /// public void Clear() { _vm = null; _initial_data = false; UpdateChart(true); } /// /// 更新图表 /// public void UpdateChart(bool calc_coordinate = false) { if (calc_coordinate || _coordinate == null) { //不强迫计算,就用上次更新的坐标系 CalcCoordinate(); } CalcSeries(); CalcChartAxis(); } #endregion #region Calc private double _min_flow, _max_flow; private double _max_head = 0, _min_head = 10000; private double _max_eff = 0, _min_eff = 0; private double _max_power = 0, _min_power = 1000; /// /// 计算坐标 /// private void CalcCoordinate() { if (_vm == null) { //设置成白板坐标 _coordinate = new PumpCoordinate(); _coordinate.GridNumberX = 30; _coordinate.GridNumberY = 16; //显示的坐标线号 _coordinate.StartLineNoH = 10; _coordinate.EndLineNoH = 15; _coordinate.StartLineNoE = 0; _coordinate.EndLineNoE = 10; _coordinate.StartLineNoP = 2; _coordinate.EndLineNoP = 9; //坐标最小值和间隔 _coordinate.CoordMinQ = 0; _coordinate.CoordSpaceQ = 1000; _coordinate.CoordMinH = 10; _coordinate.CoordSpaceH = 100; _coordinate.CoordMinE = 0; _coordinate.CoordSpaceE = 100; _coordinate.CoordMinP = 10; _coordinate.CoordSpaceP = 100; return; } _max_flow = 0; _min_flow = 10000; _max_head = 0; _min_head = 10000; _max_eff = 0; _min_eff = 0; _max_power = 0; _min_power = 1000; double _scaleMinH = 1, _scaleMaxH = 1; { var qh_pt_list = _vm.CurveQH.GetPointList(); var xxx = qh_pt_list.Select(x => x.X); var yyy = qh_pt_list.Select(x => x.Y); _min_flow = Math.Min(_min_flow, xxx.Min()); _max_flow = Math.Max(_max_flow, xxx.Max()); _min_head = Math.Min(_min_head, yyy.Min()); _max_head = Math.Max(_max_head, yyy.Max()); } if (_vm.CurveQE != null) { var qe_pt_list = _vm.CurveQE.GetPointList(); var yyy = qe_pt_list.Select(x => x.Y); _min_eff = Math.Min(_min_eff, yyy.Min()); _max_eff = Math.Max(_max_eff, yyy.Max()); } if (_vm.CurveQP != null) { var yyy = _vm.CurveQP.GetPointList().Select(x => x.Y); _min_power = Math.Min(_min_power, yyy.Min()); _max_power = Math.Max(_max_power, yyy.Max()); } if (_vm.Items != null && _vm.Items.Any()) { foreach (var vm in _vm.Items) { { var qh_pt_list = vm.CurveQH.GetPointList(); var xxx = qh_pt_list.Select(x => x.X); var yyy = qh_pt_list.Select(x => x.Y); _min_flow = Math.Min(_min_flow, xxx.Min()); _max_flow = Math.Max(_max_flow, xxx.Max()); _min_head = Math.Min(_min_head, yyy.Min()); _max_head = Math.Max(_max_head, yyy.Max()); } if (vm.CurveQE != null) { var qe_pt_list = vm.CurveQE.GetPointList(); var yyy = qe_pt_list.Select(x => x.Y); _min_eff = Math.Min(_min_eff, yyy.Min()); _max_eff = Math.Max(_max_eff, yyy.Max()); } if (vm.CurveQP != null) { var yyy = vm.CurveQP.GetPointList().Select(x => x.Y); _min_power = Math.Min(_min_power, yyy.Min()); _max_power = Math.Max(_max_power, yyy.Max()); } } } _coordinate = PumpCoordinate.CalcCoordinate(_min_flow, _max_flow, _min_head * _scaleMinH, _max_head * _scaleMaxH, _min_eff, _max_eff, _min_power, _max_power); if (_coordinate == null) return; if (_coordinate.CoordMinQ + _coordinate.CoordSpaceQ * this._coordinate.GridNumberX < _max_flow * 1.05) { _coordinate.GridNumberX++; } } /// /// 计算图表轴 /// private void CalcChartAxis() { _axis_x_flow.Visibility = DefaultBoolean.False; _axis_x_flow.GridLines.Visible = false; _axis_y_head.Visibility = DefaultBoolean.False; _axis_y_head.GridLines.Visible = false; _axis_y_eff.Visibility = DefaultBoolean.False; _axis_y_eff.GridLines.Visible = false; _axis_y_power.Visibility = DefaultBoolean.False; _axis_y_power.GridLines.Visible = false; _bottom_pane.Visibility = ChartElementVisibility.Hidden; _bottom_pane.Visibility = _qp_visible ? ChartElementVisibility.Visible : ChartElementVisibility.Hidden; //计算刻度 Q var axisQLabels = new List(); var disQ = _coordinate.CoordMinQ; for (int i = 0; i < _coordinate.GridNumberX + 1; i++) { axisQLabels.Add(new CustomAxisLabel(disQ.ToString("N0"), disQ)); disQ = disQ + _coordinate.CoordSpaceQ; } _axis_x_flow.CustomLabels.Clear(); _axis_x_flow.CustomLabels.AddRange(axisQLabels.ToArray()); _axis_x_flow.Visibility = DefaultBoolean.True; _axis_x_flow.GridLines.Visible = true; //计算刻度 var axis_head_labels = new List(); var display_head = _coordinate.CoordMinH + _coordinate.CoordSpaceH * _coordinate.StartLineNoH; for (int i = _coordinate.StartLineNoH; i < _coordinate.EndLineNoH + 1; i++) { axis_head_labels.Add(new CustomAxisLabel(display_head.ToString(), display_head)); display_head = display_head + _coordinate.CoordSpaceH; } _axis_y_head.CustomLabels.Clear(); _axis_y_head.CustomLabels.AddRange(axis_head_labels.ToArray()); _axis_y_head.Visibility = DefaultBoolean.True; _axis_y_head.GridLines.Visible = true; //效率 if (_max_eff > _min_eff && _qe_visible) { //计算刻度 var labels = new List(); var display_eff = _coordinate.CoordMinE + _coordinate.CoordSpaceE * _coordinate.StartLineNoE; for (int i = _coordinate.StartLineNoE; i < _coordinate.EndLineNoE + 1; i++) { labels.Add(new CustomAxisLabel(display_eff.ToString(), display_eff)); display_eff = display_eff + _coordinate.CoordSpaceE; } _axis_y_eff.CustomLabels.Clear(); _axis_y_eff.CustomLabels.AddRange(labels.ToArray()); _axis_y_eff.Visibility = DefaultBoolean.True; _axis_y_eff.GridLines.Visible = true; } //功率 if (_max_power > _min_power && _qp_visible) { //计算刻度 var labels = new List(); double display_power = _coordinate.CoordMinP + _coordinate.CoordSpaceP * _coordinate.StartLineNoP; for (int i = _coordinate.StartLineNoP; i < _coordinate.EndLineNoP + 1; i++) { labels.Add(new CustomAxisLabel(display_power.ToString(), display_power)); display_power = display_power + _coordinate.CoordSpaceP; } _axis_y_power.CustomLabels.Clear(); _axis_y_power.CustomLabels.AddRange(labels.ToArray()); _axis_y_power.Visibility = DefaultBoolean.True; _axis_y_power.GridLines.Visible = true; } var min_flow = _coordinate.CoordMinQ; var max_flow = _coordinate.DispMaxQ(); _axis_x_flow.SetAxisRange(min_flow, max_flow); var grid_count_head = _coordinate.EndLineNoH - _coordinate.StartLineNoH; var grid_count_eff = _coordinate.EndLineNoE - _coordinate.StartLineNoE; int grid_count_up = Math.Max(grid_count_head, grid_count_eff); if (_qe_visible) { grid_count_up += 2;//多两条 } var max_axis_head = _coordinate.CoordMinH + _coordinate.EndLineNoH * _coordinate.CoordSpaceH; var min_axis_head = max_axis_head - grid_count_up * _coordinate.CoordSpaceH; _axis_y_head.SetAxisRange(min_axis_head, max_axis_head); var min_axis_eff = _coordinate.CoordMinE + _coordinate.StartLineNoE * _coordinate.CoordSpaceE; var max_axis_eff = min_axis_eff + grid_count_up * _coordinate.CoordSpaceE; _axis_y_eff.SetAxisRange(min_axis_eff, max_axis_eff); var grid_count_power = _coordinate.EndLineNoP - _coordinate.StartLineNoP; var min_axis_power = _coordinate.CoordMinP + _coordinate.StartLineNoP * _coordinate.CoordSpaceP; var max_axis_power = min_axis_power + grid_count_power * _coordinate.CoordSpaceP; _axis_y_power.SetAxisRange(min_axis_power, max_axis_power); } /// /// 计算系列 /// private void CalcSeries() { this.chartControl1.BeginInit(); this.chartControl1.Series.Clear(); this.chartControl1.AnnotationRepository.Clear(); this.chartControl1.Legend.CustomItems.Clear(); if (_vm != null) { _vm.Color = RandomColorHelper.Get(0); CreateLineSeries(_vm.Id, _vm.Color, _vm.Name, _vm.CurveQH, _vm.CurveQE, _vm.CurveQP); if (_vm.Items != null && _vm.Items.Any()) { for (int i = 0; i < _vm.Items.Count; i++) { var vm = _vm.Items[i]; vm.Color = RandomColorHelper.Get(i + 1); CreateLineSeries(vm.Id, vm.Color, vm.CurveName, vm.CurveQH, vm.CurveQE, vm.CurveQP); } } } this.chartControl1.EndInit(); } /// /// 创建线系列 /// private void CreateLineSeries(string id, Color color, string curve_name, Yw.Geometry.CubicSpline2d qh, Yw.Geometry.CubicSpline2d qe, Yw.Geometry.CubicSpline2d qp) { var series_qh = new DevExpress.XtraCharts.Series(); series_qh.ArgumentScaleType = DevExpress.XtraCharts.ScaleType.Numerical; series_qh.LabelsVisibility = DevExpress.Utils.DefaultBoolean.False; series_qh.Name = _tag_qh + id; series_qh.ShowInLegend = false; series_qh.CrosshairEnabled = DefaultBoolean.False; series_qh.LegendTextPattern = curve_name; var series_qh_view = new DevExpress.XtraCharts.SplineSeriesView(); series_qh_view.LineStyle.Thickness = 2; series_qh_view.Color = color; series_qh_view.EnableAntialiasing = DefaultBoolean.True; series_qh_view.LineTensionPercent = 50; series_qh.SeriesPointsSorting = SortingMode.None; series_qh.SeriesPointsSortingKey = SeriesPointKey.Value_1; series_qh.View = series_qh_view; series_qh.Visible = true; var pt_qh_list = qh.GetPointList(12); for (int i = 0; i < pt_qh_list.Count; i++) { series_qh.Points.Add(new SeriesPoint(pt_qh_list[i].X, new double[] { pt_qh_list[i].Y })); } var point_qh = pt_qh_list[pt_qh_list.Count() - 1]; var anchor_qh_pt = new DevExpress.XtraCharts.PaneAnchorPoint(); anchor_qh_pt.Pane = _default_pane; anchor_qh_pt.AxisXCoordinate.AxisValue = point_qh.X.ToString(); anchor_qh_pt.AxisYCoordinate.AxisValue = point_qh.Y.ToString(); var position_qh = new DevExpress.XtraCharts.RelativePosition(); position_qh.Angle = -50; position_qh.ConnectorLength = 10; var txt_qh = new TextAnnotation(); txt_qh.Border.Visibility = DefaultBoolean.False; txt_qh.AnchorPoint = anchor_qh_pt; txt_qh.AutoHeight = true; txt_qh.AutoWidth = true; txt_qh.BackColor = System.Drawing.Color.Transparent; txt_qh.Border.Color = color; txt_qh.ConnectorStyle = DevExpress.XtraCharts.AnnotationConnectorStyle.Line; txt_qh.DXFont = PumpChartDisplay.AnnoFontQH; txt_qh.Name = _tag_qh + id; txt_qh.Padding.Bottom = 1; txt_qh.Padding.Left = 1; txt_qh.Padding.Right = 1; txt_qh.Padding.Top = 1; txt_qh.RuntimeAnchoring = false; txt_qh.RuntimeMoving = true; txt_qh.RuntimeResizing = false; txt_qh.RuntimeRotation = false; txt_qh.Text = curve_name; txt_qh.TextColor = color; txt_qh.ShapePosition = position_qh; txt_qh.Visible = true; this.chartControl1.AnnotationRepository.Add(txt_qh); this.chartControl1.Series.Add(series_qh); if (qe != null) { var series_qe = new DevExpress.XtraCharts.Series(); series_qe.ArgumentScaleType = DevExpress.XtraCharts.ScaleType.Numerical; series_qe.LabelsVisibility = DevExpress.Utils.DefaultBoolean.False; series_qe.Name = _tag_qe + id; series_qe.ShowInLegend = false; series_qe.CrosshairEnabled = DefaultBoolean.False; series_qe.Tag = id; var series_qe_view = new DevExpress.XtraCharts.SplineSeriesView(); series_qe_view.LineStyle.Thickness = 2; series_qe_view.Color = color; series_qe_view.AxisY = _axis_y_eff; series_qe_view.Pane = _default_pane; series_qe_view.EnableAntialiasing = DefaultBoolean.True; series_qe_view.LineTensionPercent = 50; series_qe.SeriesPointsSorting = SortingMode.None; series_qe.SeriesPointsSortingKey = SeriesPointKey.Value_1; series_qe.View = series_qe_view; series_qe.Visible = _qe_visible; var pt_qe_list = qe.GetPointList(12); for (int i = 0; i < pt_qe_list.Count; i++) { series_qe.Points.Add(new SeriesPoint(pt_qe_list[i].X, new double[] { pt_qe_list[i].Y })); } this.chartControl1.Series.Add(series_qe); } if (qp != null) { var series_qp = new DevExpress.XtraCharts.Series(); series_qp.ArgumentScaleType = DevExpress.XtraCharts.ScaleType.Numerical; series_qp.LabelsVisibility = DevExpress.Utils.DefaultBoolean.False; series_qp.Name = _tag_qp + id; series_qp.ShowInLegend = false; series_qp.CrosshairEnabled = DefaultBoolean.False; series_qp.Tag = id; var series_qp_view = new DevExpress.XtraCharts.SplineSeriesView(); series_qp_view.LineStyle.Thickness = 2; series_qp_view.Color = color; series_qp_view.AxisY = _axis_y_power; series_qp_view.Pane = _bottom_pane; series_qp_view.EnableAntialiasing = DefaultBoolean.True; series_qp_view.LineTensionPercent = 50; series_qp.SeriesPointsSorting = SortingMode.None; series_qp.SeriesPointsSortingKey = SeriesPointKey.Value_1; series_qp.View = series_qp_view; series_qp.Visible = _qp_visible; var pt_qp_list = qp.GetPointList(12); for (int i = 0; i < pt_qp_list.Count; i++) { series_qp.Points.Add(new SeriesPoint(pt_qp_list[i].X, new double[] { pt_qp_list[i].Y })); } this.chartControl1.Series.Add(series_qp); } } #endregion #region Right Click Menu #region Event private void barBtnSetChartAxis_ItemClick(object sender, DevExpress.XtraBars.ItemClickEventArgs e) { SetChartAxis(); } private void barCekCurveQEVisible_CheckedChanged(object sender, DevExpress.XtraBars.ItemClickEventArgs e) { _qe_visible = this.barCekCurveQEVisible.Checked; UpdateChart(); } private void barCekCurveQPVisible_CheckedChanged(object sender, DevExpress.XtraBars.ItemClickEventArgs e) { _bottom_pane.Visibility = _bottom_pane.Visibility == ChartElementVisibility.Visible ? ChartElementVisibility.Hidden : ChartElementVisibility.Visible; _qp_visible = _bottom_pane.Visibility == ChartElementVisibility.Visible ? true : false; UpdateChart(); } private void barCekCurveEQVisible_CheckedChanged(object sender, DevExpress.XtraBars.ItemClickEventArgs e) { _equip_visible = this.barCekCurveEQVisible.Checked; } #endregion /// /// 设置坐标轴 /// public void SetChartAxis() { var dlg = new PumpChartCoordinateDlg(); dlg.SetBindingData(_coordinate); dlg.OnChangedCoord += (rhs) => { _coordinate = rhs; CalcChartAxis(); this.CoordinateChangedEvent?.Invoke(_coordinate); }; dlg.ShowDialog(); } #endregion } }