在ViewModel中更改ObservableCollection的属性而不选择它

时间:2013-10-21 19:13:53

标签: c# wpf mvvm treeview observablecollection

在我的程序中,我有一个TreeView,它是通过ViewModel使用ObservableCollection实现的。每个集合都有一个名为Rank的属性。这应该作为集合项的索引。

this question中,我能够使用TreeView

让我的ObservableCollection.Move();个节点切换位置

但是,在切换节点的位置后,我需要更正/更改节点级别的值,以便我可以继续操作它们。

这应该有助于解释我在做什么:

观看 - 代码背后:

//Button Click Event -- This makes the Selected Node switch places with the node above it
private void shiftUp_Click(object sender, RoutedEventArgs e)
{
   //if a node is selected
   if (UCViewModel.TreeViewViewModel.SelectedItem != null)
   {
       //If the selected Node is not in the 0 position (can not move up anymore)
       if (UCViewModel.TreeViewViewModel.Collection<TreeViewModel>.IndexOf(UCViewModel.TreeViewViewModel.SelectedItem) != 0)
       {
           int oldIndex = UCViewModel.TreeViewViewModel.SelectedItem.Rank;
           int newIndex = oldIndex--;

           UCViewModel.TreeViewViewModel.Collection<TreeViewModel>.Move(oldIndex, newIndex);

           //**Pseudo code trying to explain what I want to do
           //**get item at specific index and change the Rank value 
           //Collection item at index (newIndex).Rank -= 1;
           //Collection item at index (oldIndex).Rank += 1;
       }
   }
}

UserControl - XAML:

<TreeView ItemsSource="{Binding TreeViewViewModel.Collection<TreeModel>}" ... />

如何在移动后更正Rank值?

修改

如上所述,我Rank的数据模型中有TreeView属性。 @Noctis的回答建议在TreeView值更改后使用该属性对Rank进行排序。我最喜欢的关于这个主题的问题here证明了这一点。

我已将SortObservableCollection类添加到我的程序中,所以现在剩下的就是操纵排名值和排序。这样做的正确位置是否来自代码隐藏?基本上上面的^部分来自哪里?如果是这样的话,我对确切的电话有点困惑......

代码隐藏:

private void shiftUp_Click(object sender, RoutedEventArgs e)
{
     //if a node is selected
     if (UCViewModel.TreeViewViewModel.SelectedItem != null)
     {
         //Moves the selectedNode down one (Up visually, hence shiftUp)
         UCViewModel.TreeViewViewModel.SelectedItem.Rank--;

         //How would I get the node below the selected one and change the Rank?

         //This would be the call to sort. Which needs to be called for the collection
         //For some reason, sort does not come up for the collection...
         //UCViewModel.TreeViewViewModel.Collection.**Sort(...);
     }
}

1 个答案:

答案 0 :(得分:1)

您可以更改对象排名上的实际值(假设它们是公共属性),而不是实际移动项目,并通过绑定自动完成排序...

修改 这很令人尴尬,正如你的评论所说,我的链接是winforms。

话虽如此,请rachelwpf-it和自我回答jeremy查看这些精彩答案。

他们要么实现了比较器(我将使用msdn链接的方向),要么使用默认标头。

这应该会给你一个很好的开始和例子,说明如何用你的等级来做。

修改

在Xaml上:

<StackPanel x:Name="LayoutRoot">
<TreeView Name="TestTreeView" ItemsSource="{Binding MyTree}">
    <TreeView.ItemTemplate>
        <DataTemplate>
            <Border BorderBrush="Black" BorderThickness="1" Margin="2" Padding="2">
            <TextBlock Text="{Binding Path=Name}"/>
            </Border>
        </DataTemplate>
    </TreeView.ItemTemplate>
</TreeView>
<Button Command="{Binding SortMe_Command}">Sort</Button>

在视图模型上,我有一个像这样的简单类(有等级,对你而言):

public class MyTreeClass
{

    public string Name { get; set; }
    public int Rank { get; set; }
}

我按原样添加了rachel的课程:

public class SortableObservableCollection<T> : ObservableCollection<T> { ...}

绑定的属性:

public SortableObservableCollection<MyTreeClass> MyTree
    {
        get { return _myTree; }
        set { _myTree = value; }
    }
private SortableObservableCollection<MyTreeClass> _myTree;

行动命令:

  

public ICommand SortMe_Command {get;组; }

在构造函数中:

MyTree = new SortableObservableCollection<MyTreeClass>() {new MyTreeClass(){Name = "One",Rank = 1},
        new MyTreeClass(){Name = "Two",Rank = 2},
        new MyTreeClass(){Name = "Three",Rank = 3}};
        SortMe_Command = new RelayCommand<object>(Execute_SortMe);
SortMe_Command = new RelayCommand<object>(Execute_SortMe);

最后但并非最不重要的是,执行方法:

private void Execute_SortMe(object obj)
    {
        MyTree[0].Rank = 5;
        MyTree[1].Rank = 4;

        MyTree.Sort(node => node.Rank);
    }

现在,当我点击按钮时,它会改变2项的等级,并根据等级求助树。

在您的应用程序中,只需确定您正在交换哪一个,并且您已经设置...