Search code examples
c#wpfdata-bindinginotifypropertychangedoxyplot

An Oxyplot graph doesn't refresh when a number in textbox changes (WPF)


I try to make a simple WPF application with MVVM pattern. It has a textBox, a textBlock, and an Oxyplot plotView. ** ViewModel class**

public class ViewModel : INotifyPropertyChanged
{

    private PlotModel? plotModel;
    public PlotModel? PlotModel
    {
        get => plotModel;
        set { plotModel = value; OnPropertyChanged("PlotModel"); }
    }

    public LineSeries lineSeries { get; set; }

    private ObservableCollection<DataPoint>? _BestFitness;
    public ObservableCollection<DataPoint>? BestFitness
    {
        get => _BestFitness;
        set { _BestFitness = value; OnPropertyChanged(nameof(BestFitness)); }
    }

    public ViewModel()
    {
        PlotModel = new PlotModel();
        plot_curve();
    }

    private double _finish;
    public double Finish
    {
        get => _finish;
        set {
            _finish = value;
            OnPropertyChanged("Finish");
            OnPropertyChanged("PlotModel");
            PlotModel?.InvalidatePlot(true);
        }
    }

    public void plot_curve() {
        Model model = new();
        PlotModel?.Series.Clear();
        lineSeries = new();
        MessageBox.Show(Finish.ToString());
        BestFitness = new();
        for (double x = 0.0; x <= 5.0; x += 0.1)
            BestFitness.Add(new DataPoint(x, model.curve(x, 2)));
        BestFitness.Add(new DataPoint(5.1, model.curve(Finish, 2)));
        lineSeries.ItemsSource = BestFitness;
        PlotModel?.Series.Add(lineSeries);
        PlotModel?.InvalidatePlot(true);
    }

    public event PropertyChangedEventHandler? PropertyChanged;
    protected virtual void OnPropertyChanged(string propertyName) => PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}

XAML

<oxy:PlotView 
    BorderBrush="Black" 
    x:Name="Plot1" 
    Model="{Binding PlotModel}"
    Margin="10,10,10,90" 
    BorderThickness="1,1,1,1" 
    Opacity="1.0" 
    Grid.ColumnSpan="3" 
    FontSize="14" 
    Grid.Column="3">
</oxy:PlotView>
<TextBox Grid.Column="2" HorizontalAlignment="Left" Margin="10,57,0,0" TextWrapping="Wrap" VerticalAlignment="Top" Width="120"
         Text="{Binding Finish, UpdateSourceTrigger=PropertyChanged, Mode=TwoWay}"/>
<TextBlock Grid.Column="2" HorizontalAlignment="Left" Margin="10,77,0,0" Text="{Binding Finish}"/>

I try to refresh the graph by adding the point named "Finish". "Finish" is being entered using the textBox. I use INotifyPropertyChanged, ObservableCollection, InvalidatePlot(true), but it doesn't work. The textBox-textBlock binding works fine, and I can see the Finish value in the textBlock. What am I missing in the code to refresh the graph?


Solution

  • if dataPoint should update when Finsih point changes, then you have to change the collection:

    private double _finish;
    public double Finish
    {
        get => _finish;
        set 
        {
            _finish = value;
            BestFitness.RemoveAt(BestFitness.Count - 1);
            BestFitness.Add(new DataPoint(5.1, model.curve(_finish, 2)));
            OnPropertyChanged("Finish");
            OnPropertyChanged("PlotModel");
            PlotModel?.InvalidatePlot(true);
        }
    }