using DevExpress.Utils;
|
using DevExpress.XtraCharts;
|
|
namespace Yw.WinFrmUI.Phart
|
{
|
/// <summary>
|
///
|
/// </summary>
|
public partial class ValveEditChart : DevExpress.XtraEditors.XtraUserControl
|
{
|
public ValveEditChart()
|
{
|
InitializeComponent();
|
InitialChart();
|
this.chartControl1.RuntimeHitTesting = true;
|
}
|
|
#region Private Variable
|
|
private XYDiagram _diagram;
|
private AxisX _axis_x_flow;
|
private AxisY _axisYQL;
|
|
private Series _seriesCurveQL;
|
private Series _seriesPointQL;
|
|
private List<Yw.Geometry.Point2d> _def_pt_ql_list;
|
|
private List<Yw.Geometry.Point2d> _pt_ql_list;
|
|
private ValveCoordinate _coordinate;
|
private bool _initial_data = false;
|
|
|
#endregion
|
|
#region Public Evnet
|
|
public event Action<ValveCoordinate> OnCurveCoordinateChanged;
|
|
public event Action<List<Yw.Geometry.Point2d>> OnDefinePointChanged;
|
|
#endregion
|
|
#region Initial
|
|
/// <summary>
|
/// 初始化图表
|
/// </summary>
|
private void InitialChart()
|
{
|
this.chartControl1.SetChartDisplay();
|
this.chartControl1.Legend.Visibility = DevExpress.Utils.DefaultBoolean.False;
|
this.chartControl1.SeriesSelectionMode = SeriesSelectionMode.Point;
|
this.chartControl1.SelectionMode = ElementSelectionMode.Single;
|
|
_diagram = (XYDiagram)chartControl1.Diagram;
|
|
|
_axis_x_flow = _diagram.AxisX;
|
_axis_x_flow.SetAxisXQDisplay();
|
_axisYQL = _diagram.AxisY;
|
_axisYQL.SetAxisYQLDisplay();
|
|
|
_seriesCurveQL = this.chartControl1.GetSeriesByName("SeriesCurveQL");
|
_seriesCurveQL.SetCurveQLDisplay();
|
|
|
_seriesPointQL = this.chartControl1.GetSeriesByName("SeriesPointQL");
|
_seriesPointQL.SetPointQLDisplay();
|
|
|
_axis_x_flow.Visibility = DefaultBoolean.False;
|
_axis_x_flow.GridLines.Visible = false;
|
_axisYQL.Visibility = DefaultBoolean.False;
|
_axisYQL.GridLines.Visible = false;
|
|
|
_seriesPointQL.Visible = false;
|
|
|
this.chartControl1.SetChartMonoColorDisplay();
|
|
|
this.chartControl1.MouseMove += new System.Windows.Forms.MouseEventHandler(this.chartControl1_MouseMove);
|
this.chartControl1.MouseUp += new System.Windows.Forms.MouseEventHandler(this.chartControl1_MouseUp);
|
this.chartControl1.MouseDown += new System.Windows.Forms.MouseEventHandler(this.chartControl1_MouseDown);
|
this.chartControl1.KeyUp += new System.Windows.Forms.KeyEventHandler(this.chartControl1_KeyUp);
|
}
|
|
|
|
/// <summary>
|
/// 初始化图表数据
|
/// </summary>
|
public void InitialChartData()
|
{
|
_initial_data = false;
|
|
_pt_ql_list = null;
|
|
_def_pt_ql_list = null;
|
|
_coordinate = null;
|
|
UpdateChart(false);
|
}
|
|
#endregion
|
|
#region SetBindingData
|
|
/// <summary>
|
/// 绑定数据(目前没有定义点)
|
/// </summary>
|
/// <param name="cubic_spline_ql"></param>
|
/// <param name="coordinate_paras"></param>
|
/// <param name="calc_coordinate"></param>
|
public void SetBindingData(Yw.Geometry.CubicSpline2d cubic_spline_ql,
|
string coordinate_paras = null,
|
bool calc_coordinate = false)
|
{
|
if (cubic_spline_ql == null)
|
{
|
InitialChartData();
|
return;
|
}
|
|
List<Yw.Geometry.Point2d> pt_ql_list;
|
|
pt_ql_list = cubic_spline_ql.GetPointList(100);
|
|
SetBindingData(pt_ql_list, pt_ql_list, coordinate_paras, calc_coordinate);
|
}
|
|
|
/// <summary>
|
/// 绑定数据
|
/// </summary>
|
/// <param name="pt_ql_list"></param>
|
/// <param name="def_pt_ql_list"></param>
|
/// <param name="coordinate_paras"></param>
|
/// <param name="calc_coordinate"></param>
|
public void SetBindingData(
|
List<Yw.Geometry.Point2d> pt_ql_list,
|
List<Yw.Geometry.Point2d> def_pt_ql_list,
|
string coordinate_paras = null,
|
bool calc_coordinate = false)
|
{
|
if (pt_ql_list == null || !pt_ql_list.Any())
|
{
|
InitialChartData();
|
return;
|
}
|
_initial_data = true;
|
|
_pt_ql_list = pt_ql_list;
|
|
_def_pt_ql_list = def_pt_ql_list;
|
|
_coordinate = ValveCoordinate.ToModel(coordinate_paras);
|
|
UpdateChart(calc_coordinate);
|
}
|
|
#endregion
|
|
#region Update
|
|
|
/// <summary>
|
/// 更新曲线
|
/// </summary>
|
/// <param name="curve_type"></param>
|
/// <param name="fit_point_list"></param>
|
public void UpdateCurve(List<Yw.Geometry.Point2d> fit_point_list)
|
{
|
if (fit_point_list == null || !fit_point_list.Any())
|
return;
|
|
_def_pt_ql_list = fit_point_list;
|
_pt_ql_list = fit_point_list.GetFitPointList();
|
_axisYQL.SetAxisYQLColorDisplay(_seriesCurveQL, _seriesPointQL, true);
|
|
UpdateChart(true);
|
}
|
|
/// <summary>
|
/// 更新图表
|
/// </summary>
|
/// <param name="calc_coordinate">计算坐标</param>
|
public void UpdateChart(bool calc_coordinate = false)
|
{
|
if (calc_coordinate || _coordinate == null)
|
{
|
//不强迫计算,就用上次更新的坐标系
|
CalcCoordinate();
|
}
|
|
CalcChartAxis();
|
|
CalcSeries();
|
}
|
|
#endregion
|
|
#region Calc
|
|
/// <summary>
|
/// 计算坐标
|
/// </summary>
|
public void CalcCoordinate()
|
{
|
if (_pt_ql_list == null || _pt_ql_list.Count < 4)
|
{
|
//设置成白板坐标
|
_coordinate = new ValveCoordinate();
|
_coordinate.GridNumberX = 30;
|
_coordinate.GridNumberY = 16;
|
//显示的坐标线号
|
_coordinate.StartLineNoL = 10;
|
_coordinate.EndLineNoL = 15;
|
//坐标最小值和间隔
|
_coordinate.CoordMinQ = 0; _coordinate.CoordSpaceQ = 1000;
|
_coordinate.CoordMinL = 10; _coordinate.CoordSpaceL = 100;
|
return;
|
}
|
|
_coordinate = ValveCoordinate.CalcCoordinate(_pt_ql_list);
|
}
|
|
/// <summary>
|
/// 计算图表轴
|
/// </summary>
|
public void CalcChartAxis()
|
{
|
if (_coordinate == null)
|
{
|
_axis_x_flow.Visibility = DefaultBoolean.False;
|
_axis_x_flow.GridLines.Visible = false;
|
_axisYQL.Visibility = DefaultBoolean.False;
|
_axisYQL.GridLines.Visible = false;
|
return;
|
}
|
|
//流量
|
if (_pt_ql_list != null)
|
{
|
//计算刻度
|
var labels = new List<CustomAxisLabel>();
|
var disQ = _coordinate.CoordMinQ;
|
for (int i = 0; i < _coordinate.GridNumberX + 1; i++)
|
{
|
labels.Add(new CustomAxisLabel(disQ.ToString("N0"), disQ));
|
disQ = disQ + _coordinate.CoordSpaceQ;
|
}
|
|
//坐标刻度
|
_axis_x_flow.CustomLabels.Clear();
|
_axis_x_flow.CustomLabels.AddRange(labels.ToArray());
|
_axis_x_flow.Visibility = DefaultBoolean.True;
|
_axis_x_flow.GridLines.Visible = true;
|
_axis_x_flow.SetAxisRange(_coordinate.CoordMinQ, _coordinate.CoordMinQ + _coordinate.GridNumberX * _coordinate.CoordSpaceQ);
|
}
|
|
//水损
|
if (_pt_ql_list != null)
|
{
|
//计算刻度
|
var labels = new List<CustomAxisLabel>();
|
var display_head = _coordinate.CoordMinL + _coordinate.CoordSpaceL * _coordinate.StartLineNoL;
|
for (int i = _coordinate.StartLineNoL; i < _coordinate.EndLineNoL + 1; i++)
|
{
|
labels.Add(new CustomAxisLabel(display_head.ToString(), display_head));
|
display_head = display_head + _coordinate.CoordSpaceL;
|
}
|
|
_axisYQL.CustomLabels.Clear();
|
_axisYQL.CustomLabels.AddRange(labels.ToArray());
|
_axisYQL.Visibility = DefaultBoolean.True;
|
_axisYQL.GridLines.Visible = true;
|
}
|
|
|
_axis_x_flow.SetAxisRange(_coordinate.CoordMinQ, _coordinate.CoordMinQ + _coordinate.GridNumberX * _coordinate.CoordSpaceQ);
|
_axisYQL.SetAxisRange(_coordinate.DispMinL(), _coordinate.DispMaxL());
|
|
}
|
|
/// <summary>
|
/// 计算系列
|
/// </summary>
|
public void CalcSeries()
|
{
|
if (_pt_ql_list != null && _pt_ql_list.Count > 3)
|
{
|
_seriesCurveQL.Visible = true;
|
_seriesCurveQL.Points.Clear();
|
foreach (var curvePoint in _pt_ql_list)
|
{
|
var seriesPoint = new SeriesPoint(curvePoint.X, curvePoint.Y);
|
_seriesCurveQL.Points.Add(seriesPoint);
|
}
|
}
|
else
|
{
|
_seriesCurveQL.Points.Clear();
|
_seriesCurveQL.Visible = false;
|
}
|
|
|
if (_def_pt_ql_list != null && _def_pt_ql_list.Any())
|
{
|
_seriesPointQL.Points.Clear();
|
for (int i = 0; i < _def_pt_ql_list.Count; i++)
|
{
|
var definePoint = _def_pt_ql_list[i];
|
var seriesPoint = new SeriesPoint(definePoint.X, definePoint.Y);
|
seriesPoint.Tag = i;
|
_seriesPointQL.Points.Add(seriesPoint);
|
}
|
_seriesPointQL.Visible = true;
|
}
|
else
|
{
|
_seriesPointQL.Visible = false;
|
}
|
|
}
|
|
#endregion
|
|
#region ChartEvent
|
|
private bool _mouseMode = false;//鼠标模式
|
private SeriesPoint _pickPoint = null;// 选中点
|
private void chartControl1_MouseDown(object sender, MouseEventArgs e)
|
{
|
if (!_initial_data)
|
return;
|
var hitInfo = chartControl1.CalcHitInfo(e.Location);
|
if (e.Button == MouseButtons.Left)
|
{
|
if (hitInfo.InSeriesPoint)
|
{
|
//if ((int)hitInfo.Series.Tag != (int)_edit_curve_type)
|
//{
|
// _pickPoint = null;
|
// return;
|
//}
|
_pickPoint = hitInfo.SeriesPoint;
|
}
|
else
|
{
|
_pickPoint = null;
|
}
|
}
|
else if (e.Button == MouseButtons.Right)
|
{
|
_pickPoint = null;
|
this.popMenuChart.ShowPopup(MousePosition);
|
}
|
}
|
private void chartControl1_MouseMove(object sender, MouseEventArgs e)
|
{
|
if (!_initial_data)
|
return;
|
if (!_mouseMode)
|
return;
|
if (_pickPoint == null)
|
return;
|
var diagramCoordinates = _diagram.PointToDiagram(e.Location);
|
var axisValue = diagramCoordinates.GetAxisValue(_axisYQL);
|
if (axisValue == null)
|
return;
|
_pickPoint.Values[0] = axisValue.NumericalValue;
|
_pickPoint.NumericalArgument = diagramCoordinates.NumericalArgument;
|
|
}
|
private void chartControl1_MouseUp(object sender, MouseEventArgs e)
|
{
|
if (!_initial_data)
|
return;
|
if (_mouseMode)
|
{
|
_pickPoint = null;
|
ChartUpdateCurve();
|
}
|
}
|
private void chartControl1_KeyUp(object sender, KeyEventArgs e)
|
{
|
if (!_initial_data)
|
return;
|
if (_pickPoint == null)
|
return;
|
double spaceX = _coordinate.CoordSpaceQ / 50;
|
double spaceY = 0;
|
|
spaceY = _coordinate.CoordSpaceL / 50;
|
|
double x = _pickPoint.NumericalArgument;
|
double y = _pickPoint.Values[0];
|
if (e.KeyCode == Keys.Up)
|
{
|
y += spaceY;
|
}
|
else if (e.KeyCode == Keys.Down)
|
{
|
y -= spaceY;
|
}
|
else if (e.KeyCode == Keys.Left)
|
{
|
x -= spaceX;
|
}
|
else if (e.KeyCode == Keys.Right)
|
{
|
x += spaceX;
|
}
|
_pickPoint.Values[0] = y;
|
_pickPoint.NumericalArgument = x;
|
ChartUpdateCurve();
|
}
|
|
/// <summary>
|
/// 更新曲线
|
/// </summary>
|
private void ChartUpdateCurve()
|
{
|
_def_pt_ql_list = _seriesPointQL.Points.Select(x => new Yw.Geometry.Point2d(x.ArgumentX.NumericalArgument, x.NumericalValue)).ToList();
|
_pt_ql_list = _def_pt_ql_list.GetFitPointList();
|
|
_seriesCurveQL.Points.Clear();
|
foreach (var curvePoint in _pt_ql_list)
|
{
|
var seriesPoint = new SeriesPoint(curvePoint.X, curvePoint.Y);
|
_seriesCurveQL.Points.Add(seriesPoint);
|
}
|
OnDefinePointChanged?.Invoke(_def_pt_ql_list);
|
|
CalcChartAxis();
|
}
|
|
#endregion
|
|
#region Right Click Menu
|
|
#region Event
|
|
|
private void barCekChartEditMode_CheckedChanged(object sender, DevExpress.XtraBars.ItemClickEventArgs e)
|
{
|
_mouseMode = this.barCekChartEditMode.Checked;
|
var text = "修改模式:鼠标 / 键盘☑";
|
if (_mouseMode)
|
{
|
text = "修改模式:鼠标☑ / 键盘";
|
}
|
_diagram.DefaultPane.Title.Text = text;
|
}
|
|
private void BarBtnCalcCoordinate_ItemClick(object sender, DevExpress.XtraBars.ItemClickEventArgs e)
|
{
|
CalcCoordinate();
|
CalcChartAxis();
|
}
|
private void barCekLegendVisible_CheckedChanged(object sender, DevExpress.XtraBars.ItemClickEventArgs e)
|
{
|
SetLegendVisible(this.barCekLegendVisible.Checked);
|
}
|
|
private void barCekSetAxisNameVisible_CheckedChanged(object sender, DevExpress.XtraBars.ItemClickEventArgs e)
|
{
|
SetAxisNameVisible(this.barCekSetAxisNameVisible.Checked);
|
}
|
|
|
|
private void barBtnSetChartAxis_ItemClick(object sender, DevExpress.XtraBars.ItemClickEventArgs e)
|
{
|
SetChartAxis();
|
}
|
|
#endregion
|
|
|
/// <summary>
|
/// 设置图例显示
|
/// </summary>
|
public void SetLegendVisible(bool visible)
|
{
|
this.chartControl1.Legend.Visibility = visible ? DefaultBoolean.True : DefaultBoolean.False;
|
}
|
|
/// <summary>
|
/// 设置轴名称显示
|
/// </summary>
|
public void SetAxisNameVisible(bool visible)
|
{
|
_axis_x_flow.Title.Visibility = visible ? DefaultBoolean.True : DefaultBoolean.False;
|
_axisYQL.Title.Visibility = visible ? DefaultBoolean.True : DefaultBoolean.False;
|
}
|
|
|
/// <summary>
|
/// 设置坐标轴
|
/// </summary>
|
public void SetChartAxis()
|
{
|
var dlg = new ValveChartCoordinateDlg();
|
dlg.SetBindingData(_coordinate);
|
dlg.OnChangedCoord += (rhs) =>
|
{
|
_coordinate = rhs;
|
CalcChartAxis();
|
this.OnCurveCoordinateChanged?.Invoke(_coordinate);
|
};
|
dlg.ShowDialog();
|
}
|
|
|
#endregion
|
}
|
}
|