wpf ObservableCollection未更新视图

rkkpypqq  于 2022-11-30  发布在  其他
关注(0)|答案(3)|浏览(199)

我刚开始使用MVVM,遇到了一个障碍,我希望有人能帮助我。我正在尝试创建一个简单的视图,有2个列表框。从第一个列表框中选择的内容将填充第二个列表框。我创建了一个类,用于存储我想绑定到的信息。
MyObject类(可观察对象只是一个实现INotifyPopertyChanged的基类)

public class MyObject : ObservableObject
{
    String _name = String.Empty;
    ObservableCollection<MyObject> _subcategories;

    public ObservableCollection<MyObject> SubCategories
    {
        get { return _subcategories; }

        set
        {
            _subcategories = value;
            RaisePropertyChanged("SubCategories");
        }
    }

    public String Name
    {
        get { return _name; }
        set
        {
            _name = value;
            RaisePropertyChanged("Name");
        }
    }

    public MyObject()
    {
        _subcategories = new ObservableCollection<EMSMenuItem>();
    }
}

在我的视图模型中,我创建了两个ObservableCollections

public ObservableCollection<EMSMenuItem> Level1MenuItems { get; set; }
public ObservableCollection<EMSMenuItem> Level2MenuItems { get; set; }

在我的ViewModel的构造函数中,我有:

this.Level1MenuItems = new ObservableCollection<EMSMenuItem>();
this.Level2MenuItems = new ObservableCollection<EMSMenuItem>();
this.Level1MenuItems = LoadEMSMenuItems("Sample.Xml");

这对于Level1项很有效,并且它们正确地显示在视图中。但是,我有一个命令,当用户单击列表框中的项时会调用该命令,它包含以下内容:

Level2MenuItems = ClickedItem.SubCategories;

由于某些原因,这不会更新第二个列表框的UI。如果我在此位置放置断点,我可以看到Level2MenuItems中存储了正确的信息。如果我编写foreach循环并将它们分别添加到Level2MenuItems集合中,则它会正确显示。
同样作为测试,我向构造函数添加了以下内容:

Level2MenuItems = Level1MenuItems[0].SubCategories;

而且更新正确。
那么,为什么代码在构造函数中或循环时能够按预期工作,而在用户单击列表框中的项时却不能呢?

u59ebvdq

u59ebvdq1#

您需要引发Level2MenuItems属性的变更告知。
而不是拥有

public ObservableCollection<EMSMenuItem> Level2MenuItems { get; set; }

您需要

private ObservableCollection<EMSMenuItem> _level2MenuItems;
public ObservableCollection<EMSMenuItem> Level2MenuItems
{
    get { return _level2MenuItems; }
    set 
     {
        _level2MenuItems = value; 
        RaisePropertyChanged(nameof(Level2MenuItems));
     }
 }

前者在构造函数中起作用的原因是绑定还没有发生。但是,由于你是通过一个在绑定之后发生的命令执行来改变引用的,你需要告诉视图它发生了改变

jc3wubiy

jc3wubiy2#

您需要使ObservableCollection中的poco类实现INotifyPropertyChanged。
示例:

<viewModels:LocationsViewModel x:Key="viewModel" />
.
.
.    
<ListView
    DataContext="{StaticResource viewModel}"
    ItemsSource="{Binding Locations}"
    IsItemClickEnabled="True"
    ItemClick="GroupSection_ItemClick"
    ContinuumNavigationTransitionInfo.ExitElementContainer="True">

    <ListView.ItemTemplate>
        <DataTemplate>
            <StackPanel Orientation="Horizontal">
                <TextBlock Text="{Binding Name}" Margin="0,0,10,0" Style="{ThemeResource ListViewItemTextBlockStyle}" />
                <TextBlock Text="{Binding Latitude, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" Style="{ThemeResource ListViewItemTextBlockStyle}" Margin="0,0,5,0"/>
                <TextBlock Text="{Binding Longitude, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" Style="{ThemeResource ListViewItemTextBlockStyle}" Margin="5,0,0,0" />
            </StackPanel>
        </DataTemplate>
    </ListView.ItemTemplate>
</ListView>

public class LocationViewModel : BaseViewModel
{
    ObservableCollection<Location> _locations = new ObservableCollection<Location>();
    public ObservableCollection<Location> Locations
    {
        get
        {
            return _locations;
        }
        set
        {
            if (_locations != value)
            {
                _locations = value;
                OnNotifyPropertyChanged();
            }
        }
    }
}

public class Location : BaseViewModel
{
    int _locationId = 0;
    public int LocationId
    {
        get
        {
            return _locationId;
        }
        set
        {
            if (_locationId != value)
            {
                _locationId = value;
                OnNotifyPropertyChanged();
            }
        }
    }

    string _name = null;
    public string Name
    {
        get
        {
            return _name;
        }
        set
        {
            if (_name != value)
            {
                _name = value;
                OnNotifyPropertyChanged();
            }
        }
    }

    float _latitude = 0;
    public float Latitude 
    { 
        get
        {
            return _latitude;
        }
        set
        {
            if (_latitude != value)
            {
                _latitude = value;
                OnNotifyPropertyChanged();
            }
        }
    }

    float _longitude = 0;
    public float Longitude
    {
        get
        {
            return _longitude;
        }
        set
        {
            if (_longitude != value)
            {
                _longitude = value;
                OnNotifyPropertyChanged();
            }
        }
    }
}

public class BaseViewModel : INotifyPropertyChanged
{
    #region Events
    public event PropertyChangedEventHandler PropertyChanged;
    #endregion

    protected void OnNotifyPropertyChanged([CallerMemberName] string memberName = "")
    {
        if (PropertyChanged != null)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(memberName));
        }
    }
}

相关问题