gpt4 book ai didi

c# - 如何将 ReactiveUI 与分层数据源( TreeView )一起使用

转载 作者:太空狗 更新时间:2023-10-29 21:53:20 24 4
gpt4 key购买 nike

我找到了一种使用 ReactiveUI 在 TreeView 中动态绑定(bind)用户控件的方法。

但是……到 HierachicalDataSource 的顶级绑定(bind)在 XAML 中而不是后面的代码中,我需要直接设置 ItemsSource 而不是根据 ReactiveUI 绑定(bind)的一般模式使用 this.OneWayBind。

所以,我的问题是:我是否遗漏了 ReactiveUI 框架中的某些内容,这些内容可以让我与 this.OneWayBind 绑定(bind)并将 HierachicalDataTemplete 移动到代码隐藏或自定义用户控件中?

特别是-是否有支持分层数据模板的 OneWayBind 的另一个重载,或者在使用它时抑制调用数据模板生成的方法?

更新我已经向我的测试项目添加了选定项和对 Expand 和 Selected 的编程支持,但我必须向 XAML 添加样式。我也想用简单的 RxUI Bind 替换它。更新了示例。

以下是关键细节:

主视图中的树控件

<TreeView Name="FamilyTree" >
<TreeView.Resources>
<Style TargetType="{x:Type TreeViewItem}">
<Setter Property="IsSelected" Value="{Binding IsSelected, Mode=TwoWay}"/>
<Setter Property="IsExpanded" Value="{Binding IsExpanded, Mode=TwoWay}"/>
</Style>
<HierarchicalDataTemplate DataType="{x:Type local:TreeItem}" ItemsSource="{Binding Children}">
<reactiveUi:ViewModelViewHost ViewModel="{Binding ViewModel}"/>
</HierarchicalDataTemplate>
</TreeView.Resources>
</TreeView>

主视图背后的代码

      public partial class MainWindow : Window, IViewFor<MainVM>
{
public MainWindow()
{
InitializeComponent();
//build viewmodel
ViewModel = new MainVM();
//Register views
Locator.CurrentMutable.Register(() => new PersonView(), typeof(IViewFor<Person>));
Locator.CurrentMutable.Register(() => new PetView(), typeof(IViewFor<Pet>));
//NB. ! Do not use 'this.OneWayBind ... ' for the top level binding to the tree view
//this.OneWayBind(ViewModel, vm => vm.Family, v => v.FamilyTree.ItemsSource);
FamilyTree.ItemsSource = ViewModel.Family;
}
...
}

主视图模型

    public class MainVM : ReactiveObject
{
public MainVM()
{
var bobbyJoe = new Person("Bobby Joe", new[] { new Pet("Fluffy") });
var bob = new Person("Bob", new[] { bobbyJoe });
var littleJoe = new Person("Little Joe");
var joe = new Person("Joe", new[] { littleJoe });
Family = new ReactiveList<TreeItem> { bob, joe };
_addPerson = ReactiveCommand.Create();
_addPerson.Subscribe(_ =>
{
if (SelectedItem == null) return;
var p = new Person(NewName);
SelectedItem.AddChild(p);
p.IsSelected = true;
p.ExpandPath();
});
}
public ReactiveList<TreeItem> Family { get; }
...
}

TreeItem 基类

public abstract class TreeItem : ReactiveObject
{
private readonly Type _viewModelType;

bool _isExpanded;
public bool IsExpanded
{
get { return _isExpanded; }
set { this.RaiseAndSetIfChanged(ref _isExpanded, value); }
}

bool _isSelected;
public bool IsSelected
{
get { return _isSelected; }
set { this.RaiseAndSetIfChanged(ref _isSelected, value); }
}

private TreeItem _parent;

protected TreeItem(IEnumerable<TreeItem> children = null)
{

Children = new ReactiveList<TreeItem>();
if (children == null) return;
foreach (var child in children)
{
AddChild(child);
}
}

public abstract object ViewModel { get; }
public ReactiveList<TreeItem> Children { get; }

public void AddChild(TreeItem child)
{
child._parent = this;
Children.Add(child);
}

public void ExpandPath()
{
IsExpanded = true;
_parent?.ExpandPath();
}
public void CollapsePath()
{
IsExpanded = false;
_parent?.CollapsePath();
}
}

人类

public class Person : TreeItem
{
public string Name { get; set; }
public Person(string name, IEnumerable<TreeItem> children = null)
: base(children)
{
Name = name;
}
public override object ViewModel => this;
}

个人 View 用户控制

<UserControl x:Class="TreeViewInheritedItem.PersonView"... >                 
<StackPanel>
<TextBlock Name="PersonName"/>
</StackPanel>
</UserControl>

Person 查看代码背后

public partial class PersonView : UserControl, IViewFor<Person>
{
public PersonView()
{
InitializeComponent();
this.OneWayBind(ViewModel, vm => vm.Name, v => v.PersonName.Text);
}
...
}

宠物和人一样工作。

完整项目在这里ReactiveUI Tree view Sample

最佳答案

我查看了 ReactiveUI 源代码,这是唯一的方法。 Bind 帮助器方法始终使用 DataTemplate 而不是 HierarchicalDataTemplate。

因此,此方法将为最顶层使用 XAML 绑定(bind),然后让您对所有 TreeView 项目使用 ReactiveUI 绑定(bind)。

我将了解如何创建拉取请求来处理这种边缘情况。

谢谢, 克里斯

关于c# - 如何将 ReactiveUI 与分层数据源( TreeView )一起使用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35371598/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com