Пользовательский класс на LiveCharts?

#c# #wpf #mvvm #livecharts

#c# #wpf #mvvm #livecharts

Вопрос:

Я пытаюсь изучить и внедрить LiveChart в приложении WPF на основе шаблона MVVM, но мне трудно понять, как его правильно реализовать. Допустим, у меня есть пользовательский класс, подобный этому. Я знаю, что вопрос немного запутанный, но мне трудно понять, как это работает, может кто-нибудь, пожалуйста, помочь мне, приведя простой пример того, как построить пользовательский класс на основе шаблона MVVM?

КЛАСС ErrorPrt

 public class ErrorPrt
{
    public ErrorPrt()
    {
        prtName = string.Empty;
        Count = -1;
    }
    public string prtName { get; set; }
    public int Count { get; set; }
}
  

ОБЪЯВЛЕНИЕ

 private string[] Labels { get; set; }

public SeriesCollection seriesCollection;
private SeriesCollection SeriesCollection
{
    get { return seriesCollection; }
    set { seriesCollection = value; OnPropertyChanged("SeriesCollection"); }
}
  

МЕТОД ОТОБРАЖЕНИЯ ДИАГРАММЫ

 public SeriesCollection dispalyChart(ErrorPrt[] err)
{
    SeriesCollection series = new SeriesCollection();

    List<int> vs1 = new List<int>();

    foreach (ErrorPrt e in err)
    {
        vs1.Add(e.Count);
    }

    series.Add(new ColumnSeries
    {
        Title = "REPORT",
        Values = new ChartValues<ErrorPrt> (err)
    });
    return series;
}

public string[] GetLabels(ErrorPrt[] err)
{
    string[] Labels = new string[err.Length];
    int j = 0;
    foreach(var e in err)
    {
        Labels[j] = e.prtName;
        j  ;
    }
    return Labels;
}
  

Редактировать

ЗНАЧЕНИЯ диаграммы ОБЪЯВЛЕНЫ КАК ГЛОБАЛЬНЫЕ ДЛЯ VIEWMODEL

 class BackupStatsViewModel : INotifyPropertyChanged
{
    //OMITTED CODE
    ChartValues<DataModel> values = new ChartValues<DataModel>();

        private void InitializeBarChartData(ErrorPrt[] arr)
        {


            for (int i = 0; i < arr.Count(); i  )
                values.Add(new DataModel() { Label = $"PRT {arr[i].prtName}", Value = arr[i].Count });
            // Initialize the DataModel items

            //for (double value = 0; value < 10; value  )
            //{
            //    values.Add(new DataModel() { Label = $"Column {value   1}", Value = value   10 });
            //}

            // Create a labels collection from the DataModel items
            this.ColumnLabels = new ObservableCollection<string>(values.Select(dataModel => dataModel.Label));
            var dataMapper = new CartesianMapper<DataModel>()
              .Y(dataModel => dataModel.Value)
              .Fill(dataModel => dataModel.Value > 15.0 ? Brushes.Red : Brushes.Green);

            this.ChartDataSets = new SeriesCollection
            {
              new ColumnSeries
              {
                Values = values,
                Configuration = dataMapper
              }
            };
        }
    }
}
  

Но в метке отображается «серия», а количество не метка..

Комментарии:

1. Какую диаграмму вы пытаетесь построить?

2. @BionicCode привет, спасибо за реплей, просто обычная гистограмма

Ответ №1:

Следующий пример представляет собой минимальный пример MVVM, который отображает ColumnSeries (гистограмму) на основе пользовательской модели данных:

DataModel.cs

 class DataModel : INotifyPropertyChanged
{
  private double value;   
  public double Value
  {
    get => this.value;
    set 
    { 
      this.value = value; 
      OnPropertyChanged();
    }
  }

  private string label;   
  public string Label
  {
    get => this.label;
    set 
    { 
      this.label = value; 
      OnPropertyChanged();
    }
  }

  public event PropertyChangedEventHandler PropertyChanged;
  protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
  {
    this.PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
  }
}
  

ViewModel.cs

 class ViewModel : INotifyPropertyChanged
{
  public SeriesCollection ChartDataSets { get; set; }
  public ObservableCollection<string> ColumnLabels { get; set; }

  public ViewModel()
  {
    InitializeBarChartData();
  }

  private void InitializeBarChartData()
  {
    // Initialize the DataModel items
    var values = new ChartValues<DataModel>();
    for (double value = 0; value < 10; value  )
    {
      values.Add(new DataModel() {Label = $"Column {value   1}", Value = value   10});
    }

    // Create a labels collection from the DataModel items
    this.ColumnLabels = new ObservableCollection<string>(values.Select(dataModel => dataModel.Label));

    // Define a data mapper, which tells the Chart how to extract data from the model
    // and how to map it to the corresponding axis. The mapper also allows 
    // to define a predicate which will be applied to color each data item (Fill, Stroke)
    var dataMapper = new CartesianMapper<DataModel>()
      .Y(dataModel => dataModel.Value)
      .Fill(dataModel => dataModel.Value > 15.0 ? Brushes.Red : Brushes.Green);

    this.ChartDataSets = new SeriesCollection
    {
      new ColumnSeries
      {
        Values = values,
        Configuration = dataMapper
      }
    };
  }

  public event PropertyChangedEventHandler PropertyChanged;
  protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
  {
    this.PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
  }
}
  

MainWindow.xaml

 <Window>
  <Window.DataContext>
    <ViewModel />
  </Window.DataContext>

  <CartesianChart Series="{Binding ChartDataSets}">
    <CartesianChart.AxisX>
      <Axis Labels="{Binding ColumnLabels}" />
    </CartesianChart.AxisX>
  </CartesianChart>
</Window>
  

Комментарии:

1. большое вам спасибо… единственная проблема в том, что он всегда показывает Series 6 как ярлык Hoover … а не мой ярлык

2. Метки извлекаются из ColumnLabels коллекции и отображаются по индексу. Опубликованный пример работает: он показывает правильную подсказку при наведении курсора мыши на столбец. Как выглядит ваш фактический код? В чем различия?

3. Спасибо. Ваш код почти идентичен приведенному примеру. Поскольку количество меток правильное (я полагаю), это должны быть значения из массива : $"PRT {arr[i].prtName}" . Переменная цикла i увеличивается правильно. Не могли бы вы, пожалуйста, проверить элементы arr ? Похоже ErrorPrt.prtName , он не возвращает ожидаемые значения. Правильно ли инициализированы эти элементы? Вам нужно отладить либо инициализацию ErrorPrt экземпляров, либо цикл, в котором вы инициализируете DataModel элементы, чтобы увидеть фактические значения всех ErrorPrt.prtName свойств.

4. Я должен упомянуть, что я представил DataModel как часть примера, потому что я не знал, что ErrorPrt это такое. В зависимости от его назначения вы можете рассмотреть возможность замены DataModel на ErrorPrt . Также обратите внимание, что в случае, если вы используете ErrorPrt в качестве источника привязки данных, он должен быть реализован INotifyPropertyChanged .