【问题标题】:ItemsControl do not update itself when ObservableCollection fires CollectionChanged当 ObservableCollection 触发 CollectionChanged 时,ItemsControl 不会自行更新
【发布时间】:2016-10-26 09:49:33
【问题描述】:

我已经阅读了很多关于这个问题的答案,但它们通常包含“你错过了 INotifyPropertyChanged”之类的内容。 我使用 MVVM light 来实现 ViewModelBase、ObservableObject 等。

查看:

<Window x:Class="BaseFlyingFigure.MainWindow"
    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:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:local="clr-namespace:BaseFlyingFigure"
    xmlns:helpers="clr-namespace:BaseFlyingFigure.Helpers"
    xmlns:i="http://schemas.microsoft.com/expression/2010/interactivity"
    xmlns:cmd="clr-namespace:GalaSoft.MvvmLight.Command;assembly=GalaSoft.MvvmLight.Platform"
    xmlns:system="clr-namespace:System;assembly=mscorlib"
    mc:Ignorable="d"
    Title="MainWindow" Height="450" Width="525"
    DataContext="{Binding MainViewModel, Source={StaticResource Locator}}">
<i:Interaction.Triggers>
    <i:EventTrigger EventName="Loaded">
        <cmd:EventToCommand Command="{Binding LoadedCommand}" />
    </i:EventTrigger>
    <i:EventTrigger EventName="PreviewKeyDown">
        <cmd:EventToCommand Command="{Binding PreviewKeyDownCommand}"
                            PassEventArgsToCommand="True" />
    </i:EventTrigger>
</i:Interaction.Triggers>
<Grid>
    <Grid.RowDefinitions>
        <RowDefinition Height="Auto" />
        <RowDefinition Height="*" />
    </Grid.RowDefinitions>
    <Menu Grid.Row="0">
        <MenuItem Header="File">
            <MenuItem Header="Exit" Command="{Binding AppExitCommand}" />
        </MenuItem>
    </Menu>
    <ItemsControl Grid.Row="1" helpers:SizeObserver.Observe="True"
                  helpers:SizeObserver.ObservedWidth="{Binding CanvasWidth, Mode=OneWayToSource}"
                  helpers:SizeObserver.ObservedHeight="{Binding CanvasHeight, Mode=OneWayToSource}"
                  ItemsSource="{Binding Elements, Converter={helpers:ElementToShapeConverter}, 
        Mode=OneWay, UpdateSourceTrigger=PropertyChanged}"
                  >
        <ItemsControl.ItemsPanel>
            <ItemsPanelTemplate>
                <Canvas ClipToBounds="True" />
            </ItemsPanelTemplate>
        </ItemsControl.ItemsPanel>
    </ItemsControl>
</Grid>

视图模型:

using System.Collections.ObjectModel;
using System.Diagnostics;
using System.Windows;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Shapes;
using BaseFlyingFigure.Services.Interfaces;
using GalaSoft.MvvmLight;
using GalaSoft.MvvmLight.Command;

namespace BaseFlyingFigure.ViewModels
{
    public class MainViewModel : ViewModelBase
    {
        private readonly IFigureRepository _repository;
        private ObservableCollection<Element> _elements = new ObservableCollection<Element>();


        public MainViewModel(IFigureRepository repository)
        {
            _repository = repository;

            AppExitCommand = new RelayCommand(Exit);
            LoadedCommand = new RelayCommand(WindowLoaded);

            PreviewKeyDownCommand = new RelayCommand<KeyEventArgs>(PreviewKeyDown);

            Elements.Add(new Element(new Ellipse {
                    Fill = Brushes.HotPink,
                    Width = 100,
                    Height = 100
                })
                {Left = 250, Top = 250});
        }

        public RelayCommand AppExitCommand { get; private set; }
        public RelayCommand LoadedCommand { get; private set; }
        public RelayCommand<KeyEventArgs> PreviewKeyDownCommand { get; private set; }

        public double CanvasWidth { get; set; }
        public double CanvasHeight { get; set; }

        public ObservableCollection<Element> Elements
        {
            get { return _elements; }
            set 
            {
                if (value != _elements)
                    Set(ref _elements, value);
            }
        }

        private void PreviewKeyDown(KeyEventArgs e)
        {
            switch (e.Key) {
                case Key.OemPlus:
                    Elements.Add(new Element(new Ellipse
                    {
                            Fill = Brushes.HotPink,
                            Width = 100,
                            Height = 100
                        })
                        {Left = 250, Top = 250});
                    Debug.WriteLine("+");
                    break;
            }
        }

        private void WindowLoaded()
        {
            Elements.CollectionChanged += (sender, args) => Debug.WriteLine("changed");
        }

        private void Exit() => Application.Current.Shutdown();
    }
}

转换器:

using System;
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Windows.Data;
using System.Windows.Markup;
using BaseFlyingFigure.ViewModels;

namespace BaseFlyingFigure.Helpers
{
    public class ElementToShapeConverter : MarkupExtension, IValueConverter
    {
        private static ElementToShapeConverter _converter;

        public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
        {
            var list = (value as ICollection<Element>)?.Select(el => el.Shape).ToList();
            return list;
        }

        public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
        {
            return null;
        }

        public override object ProvideValue(IServiceProvider serviceProvider)
        {
            return _converter ?? (_converter = new ElementToShapeConverter());
        }
    }
}

元素:

using System.Windows.Controls;
using System.Windows.Shapes;
using GalaSoft.MvvmLight;

namespace BaseFlyingFigure.ViewModels
{
    public class Element : ObservableObject
    {
        private double _left;
        private Shape _shape;
        private double _top;

        public Element(Shape shape)
        {
            Shape = shape;
        }

        public double Left
        {
            get { return _left; }
            set
            {
                Set(ref _left, value);
                Canvas.SetLeft(Shape, value);
            }
        }

        public double Top
        {
            get { return _top; }
            set
            {
                Set(ref _top, value);
                Canvas.SetTop(Shape, value);
            }
        }

        public Shape Shape
        {
            get { return _shape; }
            set { Set(ref _shape, value); }
        }
    }
}

当我们按下 + 时,CollectionChanged 会触发。但是画布显示在构造函数中制作和添加的形状。 ViewModelLocator:

public class ViewModelLocator
{
    public ViewModelLocator()
    {
        ServiceLocator.SetLocatorProvider(() => SimpleIoc.Default);
        SimpleIoc.Default.Register<MainViewModel>();
        SimpleIoc.Default.Register<IFigureRepository, FigureRepository>();
    }

    public MainViewModel MainViewModel => ServiceLocator.Current.GetInstance<MainViewModel>();
}

这有什么问题? XAML 或其他内容中是​​否有任何错误?

【问题讨论】:

  • 请注意,在 ItemsSource 绑定上设置 Mode=OneWayUpdateSourceTrigger=PropertyChanged 是没有意义的。 OneWay 是默认模式,UpdateSourceTrigger 只影响 TwoWay(和 OneWayToSource)绑定。
  • 此外,您不应在 ItemsSource 绑定上使用转换器。相反,将 ItemsControl 的 ItemContainerStyle 和/或 ItemTemplate 属性设置为可视化形状元素的 Stye/DataTemplate。
  • ElementToShapeConverter 会破坏正常行为。它用List(不能)替换ObservableCollection(可以通知UI)
  • @Clemens 我认为说 OneWas 是默认设置是一种误导。它并不总是默认的。 MSDN 说 BindingMode 值之一。默认为 Default,它返回目标依赖属性的默认绑定模式值。但是,每个依赖属性的默认值会有所不同。
  • @3615 当然可以,但它是 ItemsSource 的(有效)默认值。

标签: c# wpf mvvm binding


【解决方案1】:

您的方法的基本问题是您在视图模型中使用Shape 对象。

除了不可能有多个视图来可视化此视图模型(因为 UIElements 只能有一个父级),它还迫使您使用一种非常规且有缺陷的方法将 ObservableCollection&lt;Element&gt; 转换为List&lt;Shape&gt; 在 ItemsSource 绑定的转换器中。正如 cmets 和其他答案中所述,从您的转换器返回的 List&lt;Shape&gt; 不会通知视图有关 ObservableCollection&lt;Element&gt; 的更改。

适当的 MVVM 方法将使用没有 UI 元素的形状表示,例如像这样:

public class Element
{
    public Geometry Shape { get; set; }
    public Brush Fill { get; set; }
    public Brush Stroke { get; set; }
    public double StrokeThickness { get; set; }
}

您现在可以为 ItemsControl 中的形状的可视化声明一个常规 DataTemplate:

<ItemsControl ItemsSource="{Binding Elements}">
    <ItemsControl.ItemsPanel>
        <ItemsPanelTemplate>
            <Canvas/>
        </ItemsPanelTemplate>
    </ItemsControl.ItemsPanel>
    <ItemsControl.ItemTemplate>
        <DataTemplate>
            <Path Data="{Binding Shape}"
                  Fill="{Binding Fill}"
                  Stroke="{Binding Stroke}"
                  StrokeThickness="{Binding StrokeThickness}"/>
        </DataTemplate>
    </ItemsControl.ItemTemplate>
</ItemsControl>

将示例椭圆添加到您的视图模型现在看起来像这样:

Elements.Add(new Element
{
    Shape = new EllipseGeometry(new Point(250, 250), 50, 50),
    Fill = Brushes.HotPink
});

如果出于任何原因您还需要为每个元素添加额外的 x/y 位置偏移量,您可以在 Element 类中添加两个属性

public class Element
{
    ...
    public double X { get; set; }
    public double Y { get; set; }
}

并将ItemsContainerStyle 添加到使用这些属性的 ItemsControl:

<ItemsControl ItemsSource="{Binding Elements}">
    ...
    <ItemsControl.ItemContainerStyle>
        <Style TargetType="ContentPresenter">
            <Setter Property="Canvas.Left" Value="{Binding X}"/>
            <Setter Property="Canvas.Top" Value="{Binding Y}"/>
        </Style>
    </ItemsControl.ItemContainerStyle>
</ItemsControl>

【讨论】:

  • 具有X、Y属性的元素是否需要实现INotifyPropertyChanged,如果我想在元素更改属性时通知视图?
  • 是的,所有这些属性都应该通知更改(当然只有在发生任何更改时)。为简洁起见,我将其省略了。
【解决方案2】:

我认为这是因为 ElementToShapeConverter.Convert 创建了一个新的 List 并返回它。这可能不会破坏与视图模型中集合的绑定,但视图模型中的集合将不再因更改而失效。

我认为您的 ViewModel 应该有一个名为 Shapes 的单独 ObservableCollection 属性,当调用它而不是 Converter 时,它将 Shapes 从 Elements 集合中过滤掉。您可以在 Elements 集合的 CollectionChanged 事件中使其无效。

【讨论】:

  • 它实际上并没有破坏 Binding,但由于它返回一个 List,因此对源 ObservableCollection 所做的进一步更改将被忽略。
  • @Clemens:好的,你是对的。我已经根据您的反对调整了我的答案:-)
猜你喜欢
  • 2021-11-26
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-04-09
  • 1970-01-01
  • 2018-07-05
相关资源
最近更新 更多