Oxyplot 中的仿射变换
Affine transformations in Oxyplot
我想对 LineSeries
执行简单的仿射变换
不幸的是我不能简单地这样做:
foreach (var point in myLineSeries.Points)
{
point.X = point.X * Math.Cos(angle_rad) + point.Y ...
point.Y = ...
}
...因为 DataPoint
的属性 X
和 Y
没有 public set
访问器
有没有一种方法可以在 LineSeries
上执行此类操作(简单 rotations/translations)而无需调用 Clear
然后 Add
新转换 DataPoints
(这对我来说似乎不太优雅)
与其直接在 LineSeries
中操作点,不如使用 MVVM:在 View Model 中进行计算,然后将结果绑定到 查看.
ViewModel:
public class MyViewModel
{
public ObservableCollection<DataPoint> Data1 { get; set; }
public ObservableCollection<DataPoint> Data2 { get; set; }
public MyViewModel()
{
Data1 = new ObservableCollection<DataPoint>();
Data2 = new ObservableCollection<DataPoint>();
double pi = Math.PI;
double a = pi / 4; // rotation angle
double fw = 5; // wave frequency
double fs = 100 * fw; // sampling rate
double te = 1; // end time
int size = (int)(fs * te);
// do your calculations
var x1 = Enumerable.Range(0, size).Select(p => p / fs).ToArray();
var y1 = x1.Select(p => Math.Sin(2*pi*fw*p)).ToArray();
//populate your Model (original data)
for (int i = 0; i < y1.Length; i++)
Data1.Add(new DataPoint(x1[i], y1[i]));
// transform original data
var x2 = Data1.Select(p => p.X * Math.Cos(a) - p.Y * Math.Sin(a)).ToArray();
var y2 = Data1.Select(p => p.X * Math.Sin(a) + p.Y * Math.Cos(a)).ToArray();
// populate your Model (transformed data)
for (int i = 0; i < y2.Length; i++)
Data2.Add(new DataPoint(x2[i], y2[i]));
}
}
XAML:
<Window x:Class="WpfApp93.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:oxy="http://oxyplot.org/wpf"
xmlns:local="clr-namespace:WpfApp93"
mc:Ignorable="d"
Title="MainWindow" Height="350" Width="350">
<Window.DataContext>
<local:MyViewModel/>
</Window.DataContext>
<Grid>
<oxy:Plot>
<oxy:LineSeries ItemsSource="{Binding Data1}"
DataFieldX="X"
DataFieldY="Y"/>
<oxy:LineSeries ItemsSource="{Binding Data2}"
DataFieldX="X"
DataFieldY="Y"/>
</oxy:Plot>
</Grid>
</Window>
我想对 LineSeries
不幸的是我不能简单地这样做:
foreach (var point in myLineSeries.Points)
{
point.X = point.X * Math.Cos(angle_rad) + point.Y ...
point.Y = ...
}
...因为 DataPoint
的属性 X
和 Y
没有 public set
访问器
有没有一种方法可以在 LineSeries
上执行此类操作(简单 rotations/translations)而无需调用 Clear
然后 Add
新转换 DataPoints
(这对我来说似乎不太优雅)
与其直接在 LineSeries
中操作点,不如使用 MVVM:在 View Model 中进行计算,然后将结果绑定到 查看.
ViewModel:
public class MyViewModel
{
public ObservableCollection<DataPoint> Data1 { get; set; }
public ObservableCollection<DataPoint> Data2 { get; set; }
public MyViewModel()
{
Data1 = new ObservableCollection<DataPoint>();
Data2 = new ObservableCollection<DataPoint>();
double pi = Math.PI;
double a = pi / 4; // rotation angle
double fw = 5; // wave frequency
double fs = 100 * fw; // sampling rate
double te = 1; // end time
int size = (int)(fs * te);
// do your calculations
var x1 = Enumerable.Range(0, size).Select(p => p / fs).ToArray();
var y1 = x1.Select(p => Math.Sin(2*pi*fw*p)).ToArray();
//populate your Model (original data)
for (int i = 0; i < y1.Length; i++)
Data1.Add(new DataPoint(x1[i], y1[i]));
// transform original data
var x2 = Data1.Select(p => p.X * Math.Cos(a) - p.Y * Math.Sin(a)).ToArray();
var y2 = Data1.Select(p => p.X * Math.Sin(a) + p.Y * Math.Cos(a)).ToArray();
// populate your Model (transformed data)
for (int i = 0; i < y2.Length; i++)
Data2.Add(new DataPoint(x2[i], y2[i]));
}
}
XAML:
<Window x:Class="WpfApp93.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:oxy="http://oxyplot.org/wpf"
xmlns:local="clr-namespace:WpfApp93"
mc:Ignorable="d"
Title="MainWindow" Height="350" Width="350">
<Window.DataContext>
<local:MyViewModel/>
</Window.DataContext>
<Grid>
<oxy:Plot>
<oxy:LineSeries ItemsSource="{Binding Data1}"
DataFieldX="X"
DataFieldY="Y"/>
<oxy:LineSeries ItemsSource="{Binding Data2}"
DataFieldX="X"
DataFieldY="Y"/>
</oxy:Plot>
</Grid>
</Window>