Привязка к ViewModel в ListView ничего не показывает, но отлично работает с другими элементами управления - PullRequest
0 голосов
/ 28 мая 2020

Я пытаюсь привязать ObservableCollection, находящуюся в моем классе ViewModel, к ListView, который использует GridView для отображения строк, но я не могу показать свои данные в списке.

В целях тестирования я помещаю TextBlock и может успешно отображать данные с помощью Binding (однако, только если я сначала сделаю DataContext = ViewModel в коде позади, выполнение ViewModel.MyData в XAML не сработает)

Я заметил, что все, что ItemsSource связано до, в списке всегда отображается случайное количество пустых элементов, я могу сказать это, поскольку, когда я наводю указатель мыши на ListView, строки выделяются. Это число не соответствует вместимости моей коллекции.

Если в коде позади я вручную установил ItemsSource для коллекции, которую я хочу отобразить, тогда отображается правильное количество элементов, но данных по-прежнему нет отображается.

XAML

<Page x:Class="GestionStockWPF.MainPage"
      xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
      xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
      xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
      xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
      xmlns:local="clr-namespace:GestionStockWPF"
      mc:Ignorable="d" 
      d:DesignHeight="450" d:DesignWidth="800"
      Title="MainPage"
      ShowsNavigationUI="False">

    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="13*"/>
            <RowDefinition Height="77*"/>
        </Grid.RowDefinitions>

        <!-- The Binding here seems to do nothing, there isn't the correct amount of rows in the app -->

        <ListView x:Name="listView"
                  HorizontalAlignment="Stretch"
                  Grid.Row="1" Margin="30, 0, 30, 30" 
                  ItemsSource="{Binding Source=Tests}">
            <ListView.View>
                <GridView>
                    <GridViewColumn DisplayMemberBinding="{Binding Path=A}"
                        Header="Asset Number" Width="200"/>
                </GridView>
            </ListView.View>
        </ListView>

        <!-- This textblock is just here for testing purposes, it successfully shows "A" -->

        <TextBlock x:Name="textBlock" Text="{Binding Path=Tests[0].A}"
                   HorizontalAlignment="Left" Margin="252,28,0,0" TextWrapping="Wrap"
                   VerticalAlignment="Top"/>
    </Grid>
</Page>

Код за файлом

using System.Windows;
using System.Windows.Controls;

namespace GestionStockWPF
{
    public partial class MainPage : Page
    {
        public MainPageViewModel ViewModel;
        public MainPage()
        {
            InitializeComponent();
            ViewModel = new MainPageViewModel();

            // If I do DataContext = this;
            // and then set the binding of the TextBlock to "ViewModel.Tests[0].A" nothing is shown
            DataContext = ViewModel;

            // If I don't do that the listview does not contain the correct number of rows
            listView.ItemsSource = ViewModel.tests;
        }
    }
}

ViewModel и тестовый класс

using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Runtime.CompilerServices;

namespace GestionStockWPF
{
    // Not sure if I have to implement INotifyPropertyChanged but my real class does it
    public class Test : INotifyPropertyChanged
    {
        public string A { get { return "A"; } }
        public string B { get { return "B"; } }

        public event PropertyChangedEventHandler PropertyChanged;

        protected void OnPropertyChanged([CallerMemberName] string name = null)
        {
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name));
        }
    }
    public class MainPageViewModel
    {
        public ObservableCollection<Test> Tests { get; set; }
        public MainPageViewModel()
        {
            tests = new ObservableCollection<Test>();
            for (int i = 0; i < 50; ++i)
            {
                Tests.Add(new Test());
            }
        }
    }
}

1 Ответ

1 голос
/ 28 мая 2020

Ваша привязка неверна.

Это должно быть:

ItemsSource="{Binding tests}"

Вам необходимо указать путь в привязке, а не источник привязки.

Явный Часть Path не является обязательной в привязке, поэтому вы также можете изменить привязку текстового блока на:

Text="{Binding tests[0].A}"

Кстати, свойства должны использовать PascalCase, поэтому лучше называть его Tests.

...