滚动时 ListBox 抛出 ItemControl Text Visualizer 错误

ListBox throws ItemControl Text Visualizer error when scrolling

我有一个列表框,代码如下,其中填充了 900 个项目。当我快速向下滚动列表时,我得到 'An ItemsControl is inconsistent with its items source' 的异常。关于导致问题的原因有什么想法吗?

                <ListBox x:Name="FileList" ItemsSource="{Binding Files}" ScrollViewer.VerticalScrollBarVisibility="Auto" ScrollViewer.CanContentScroll="True" 
                     MaxHeight="520" >
                <ListBox.ItemTemplate>
                    <DataTemplate>
                        <Grid>
                            <Grid.ColumnDefinitions>
                                <ColumnDefinition Width="Auto" />
                                <ColumnDefinition Width="Auto" />
                                <ColumnDefinition Width="Auto" />
                                <ColumnDefinition Width="Auto" />
                                <ColumnDefinition Width="Auto" />
                            </Grid.ColumnDefinitions>
                            <CheckBox x:Name="FileChecked" IsChecked="{Binding Checked}" Grid.Column="0" BorderThickness="1"/>
                            <Label Content="{Binding Name}" Grid.Column="1" />
                            <Label Content="{Binding Source}" Grid.Column="2" Style="{StaticResource FileProperties}" />
                            <Label Content="{Binding Destination}" Grid.Column="3" Style="{StaticResource FileProperties}"/>
                            <Label Content="{Binding ArchiveLocation}" Grid.Column="4" Style="{StaticResource FileProperties}"/>
                        </Grid>
                    </DataTemplate>
                </ListBox.ItemTemplate>
            </ListBox>

enter image description here

这是该问题的视频。 https://www.screencast.com/t/YUlp24zoXiG

我相信我已经找到问题所在。我在下面粘贴了有问题的代码,以便其他访问者可以从这个答案中受益。

// Your code from GitHub unchanged
private async void Preview_Click(object sender, System.Windows.RoutedEventArgs e)
{
    vm.Files = new List<InstrumentFile>();

    try
    {
        for (int i = 0; i < InstrumentList.Items.Count; i++)
        {
            ListBoxItem lbi = (InstrumentList.ItemContainerGenerator.ContainerFromIndex(i)) as ListBoxItem;
            ContentPresenter cp = GetFrameworkElementByName<ContentPresenter>(lbi);
            DataTemplate dt = InstrumentList.ItemTemplate;
            CheckBox cb = (dt.FindName("InstrumentChecked", cp)) as CheckBox;

            if (cb.IsChecked == true)
            {
                List<InstrumentFile> instrumentFiles = new List<InstrumentFile>();
                Instrument instrument = ((Instrument)(InstrumentList.Items[i]));

                string[] files = Directory.GetFiles(instrument.FileSource);
                foreach (string file in files)
                {
                    FileInfo fi = new FileInfo(file);
                    instrumentFiles.Add(new InstrumentFile()
                    {
                        Name = fi.Name,
                        Source = instrument.FileSource,
                        Destination = instrument.Destination,
                        ArchiveLocation = instrument.ArchiveLocation
                    });
                }
                if (string.IsNullOrEmpty(instrument.FileExt) == false)
                {
                    IEnumerable<InstrumentFile> filteredFiles = instrumentFiles.Where(f => f.Name.ToUpper().EndsWith(instrument.FileExt.ToUpper()));
                    if (filteredFiles.Count() > 0)
                        vm.Files.AddRange(filteredFiles);
                }
                else
                {
                    if (instrumentFiles.Count > 0)
                        vm.Files.AddRange(instrumentFiles);
                }

            }
        }
    }
    catch (Exception ex)
    {
        await metroWindow.ShowMessageAsync("Exception Encountered", ex.Message, MessageDialogStyle.Affirmative, Helpers.DialogSettings());
    }
    FileCount.Content = vm.Files.Count.ToString() + " files";
}

在这里,您要在视图模型中初始化 Files 属性。这会导致数据绑定更新为空列表。目前没有问题。但是,您随后将内容添加到 Files,但这些更改不会传播到数据绑定系统,因为视图模型中的列表不是 ObservableCollection.

您可以通过多种方式解决此问题。一种修复方法是在 视图模型中设置 Files 属性 在您创建并填充列表后 。新代码如下所示(缩写):

private async void Preview_Click(object sender, System.Windows.RoutedEventArgs e)
{
    // Create new variable to store the list
    var files = new List<InstrumentFile>();

    // You do a bunch of things, but you now add items to files, not to vm.Files
    files.AddRange(filteredFiles);

    // Finally, change Files
    vm.Files = files
}

最后一行将引发视图模型中的 PropertyChanged 事件以更新数据绑定,但会使用完整的项目列表。

第二种修复方法是将视图模型中 Files 的类型更改为 ObservableCollection<InstrumentFile>。每当您对 Files 进行更改时,都会引发正确的事件以更新数据绑定。