ListView в UWP имеет неожиданное поведение, когда INotifyCollectionChanged сообщает о массовых изменениях

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

1) Откройте Visual Studio 2015 и создайте пустое универсальное приложение для Windows, используя Build 14393, и назовите его "SampleApp", чтобы у нас было одинаковое пространство имен.

2) Скопируйте следующий код в Main.xaml

<Page x:Class="SampleApp.MainPage"
      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:local="using:SampleApp"
      xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
      mc:Ignorable="d">

    <Grid Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
        <ListView ItemsSource="{x:Bind Items}" Margin="40"/>
    </Grid>
</Page>

3) Скопируйте следующий код в Main.xaml.cs:

using System.Collections.Generic;
using System.Collections.Specialized;
using System.ComponentModel;
using System.Threading.Tasks;
using Windows.UI.Xaml;

namespace SampleApp
{
    public sealed partial class MainPage
    {
        public FakeCollection Items { get; } = new FakeCollection();

        public MainPage()
        {
            InitializeComponent();

            Loaded += OnLoaded;
        }

        private async void OnLoaded(object sender, RoutedEventArgs e)
        {
            await Items.Begin();
        }
    }

    public class FakeCollection : List<string>, INotifyCollectionChanged, INotifyPropertyChanged
    {
        public event NotifyCollectionChangedEventHandler CollectionChanged;
        public event PropertyChangedEventHandler PropertyChanged;


        public async Task Begin()
        {
            var set1 = CreateSet(1, 10);
            var set2 = CreateSet(11, 15);

            AddRange(set1);

            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Count"));
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Item[]"));
            CollectionChanged?.Invoke(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, set1, 0));


            await Task.Delay(2000);

            InsertRange(5, set2);

            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Count"));
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Item[]"));
            CollectionChanged?.Invoke(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, set2, 5));
        }

        public List<string> CreateSet(int start, int end)
        {
            var list = new List<string>();

            for (var i = start; i <= end; i++)
            {
                list.Add($"Item {i}");
            }

            return list;
        }
    }
}

FakeCollection это просто List<string> и реализует INotifyPropertyChanged а также INotifyCollectionChanged, Используется в ListView переплет в xaml.

Когда главная страница загружается, Begin() вызывается метод, который просто создает список строк с именами "Item 1", "Item 2", ... "Item 10", добавляет их в список и затем уведомляет об изменении коллекции по всем 10 элементам. Это отлично работает.

Однако после ожидания 2 секунд создается другой набор "Элемент 11", "Элемент 12", ... "Элемент 15", и они вставляются с индексом 5. Мы сообщаем об изменении коллекции снова в позиции 5 с новым набором. струн.

Я бы ожидал ListView чтобы показать точный порядок, найденный во внутренней коллекции FakeCollection, который:

Элемент 1,2,3,4,5,11,12,13,14,15,6,7,8,9,10

Вместо этого это проявляется как

Элемент 1,2,3,4,5,11,6,7,8,9,10,7,8,9,10????

Что здесь случилось? Похоже, что он добавил первый элемент из поднятого события, а затем просто повторил последние четыре!

Есть ли хитрость, чтобы сделать ListView работать с массовыми изменениями коллекции?

1 ответ

Интересно! Похоже CollectionChanged не работает, когда вы пытаетесь добавить несколько элементов одновременно. Я думаю, вам придется сделать что-то вроде этого -

for (var i = 0; i < set2.Count; i++)
{
    CollectionChanged?.Invoke(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, set2, i + 5));
}
Другие вопросы по тегам