调用 WPF 选择 TreeView 项作为上下文菜单
Posted
技术标签:
【中文标题】调用 WPF 选择 TreeView 项作为上下文菜单【英文标题】:WPF selection of a TreeView item as context menu is called 【发布时间】:2017-08-20 02:52:50 【问题描述】:这个问题主要与上下文菜单有关,但在我的具体情况下,它与 TreeView 控件有关。
TreeView 项目包含一个 StackPanel,在该 StackPanel 上是一个 ContextMenu 属性,我已将其分配给一个 StaticResource(当然是一个 ContextMenu)。说 ContextMenu 导致一个 ICommand ,因此,做它的事。
目前(我相信这是默认行为),右键单击 TreeView 中的项目不会选择该项目。这在 Windows 中很常见,但在这里不会发生。我希望它发生(但我不知道如何)。
一些后续信息:我在 TreeView 中确实有一个选定的项目,这会通过鼠标左键单击而改变。不过,这不是左键单击事件,而是“SelectedItemChanged”事件。这导致我将数据上下文(视图模型)中的“SelectedItem”设置为 SelectedItem 的方法。必须这样做,因为 TreeView 的选定项是“只读”的。
该代码在这里,虽然我不确定它对手头的问题有多大用处:
private void TreeView_SelectedItemChanged(object sender, RoutedPropertyChangedEventArgs<object> e)
if (this.ScenesTreeView01 == null)
return;
if (this.ScenesTreeView01.DataContext == null)
return;
var DataContext = this.ScenesTreeView01.DataContext as ScenesViewModel;
if (e.NewValue is SceneViewModel)
DataContext.SelectedScene = (SceneViewModel)e.NewValue;
if (e.NewValue is CharacterViewModel)
DataContext.SelectedCharacter = (CharacterViewModel)e.NewValue;
由于似乎没有一个地方显示“好的,你左键单击,所以这里是选定的项目”,我不知道该怎么做才能告诉它在右键单击时分配选定的项目(以及左键单击)。
我该怎么做?
编辑:我使用的是 MVVM,所以当我们有一个像 SelectedItemChanged 这样带有一个参数为 RoutedPropertyChangedEventArgs e 的方法时,e.Source 将我引回我的视图模型,而不是 TreeViewItem。
【问题讨论】:
【参考方案1】:您可以将IsSelected
属性添加到SceneViewModel
和CharacterViewModel
类,并使用样式将TreeViewItem
的IsSelected
绑定到这些属性。然后,您可以以相同的方式为PreviewMouseRightButtonDown
连接一个事件处理程序来设置源属性:
<TreeView x:Name="treeView">
<TreeView.Resources>
<Style TargetType="TreeViewItem">
<Setter Property="IsSelected" Value="Binding IsSelected, Mode=TwoWay" />
<EventSetter Event="PreviewMouseRightButtonDown" Handler="treeView_PreviewMouseRightButtonDown" />
<Setter Property="ContextMenu">
<Setter.Value>
<ContextMenu>
<MenuItem>1</MenuItem>
</ContextMenu>
</Setter.Value>
</Setter>
</Style>
</TreeView.Resources>
</TreeView>
private void treeView_PreviewMouseRightButtonDown(object sender, MouseButtonEventArgs e)
TreeViewItem tvi = sender as TreeViewItem;
CharacterViewModel cvm = tvi.DataContext as CharacterViewModel;
if (cvm != null)
cvm.IsSelected = true;
else
SceneViewModel svm = tvi.DataContext as SceneViewModel;
if (svm != null)
svm.IsSelected = true;
确保CharacterViewModel
和SceneViewModel
类实现INotifyPropertyChanged
接口并在新IsSelected
属性的设置器中引发PropertyChanged
事件。
【讨论】:
我想我会详细说明,因为我说这在其他线程中对我不起作用。我将代码的第一部分放入现有的 TreeView.Resource 标记中。在后面的代码中,我也有您提到的方法。它不会触发,但由于连接在 XAML 中,我不知道如何检查发生的情况以确定它为什么不触发。我也不明白你为什么在那里有一个上下文菜单。这是否需要,因为我已经有一些上下文菜单并且它们工作正常。 嘿伙计,快速更新,因为我已经研究了很长时间了。似乎它对我不起作用的原因是因为我在我的 XAML 中已经有了该类型的样式,并且不知道您不能只拥有另一个样式。所以我将你的部分放入我现有的部分中并且它工作(所以感谢你),除了某种原因,该方法在***项目('场景')的情况下触发一次,如果我单击一个子级别项目(“角色”)。所以谢谢你让我这么远 关于多次通话问题:在这种情况下。使用MouseRightButtonDown
可能比PreviewMouseRightButtonDown
更好,因为前者会沿着树向下移动,使第一个来电成为误报,而后者将在正确的TreeViewItem
上进行第一个呼叫。还要设置e.Handled=true
,以避免事件向上传播...【参考方案2】:
一个快速的解决方案可能是简单地注册 TreeView 的 MouseRightButtonDown 事件,检查点击是否在 TreeViewItem 上并选择它:
TreeView.MouseRightButtonDown += Tv_MouseRightButtonDown;
void Tv_MouseRightButtonDown(object sender, MouseButtonEventArgs e)
var tvItem = e.Source as TreeViewItem;
if (tvItem != null)
tvItem.IsSelected = true;
【讨论】:
对不起。我添加了一个编辑来解释为什么这对我不起作用。由于我使用的是 MVVM,因此 e.Source 将我引用到来自我的 ViewModel 的一些数据对象,而不是 TreeViewItem。应该在我的问题中包含这一点,所以我再次为不清楚的道歉。 您是否仅针对 SelectedItemChanged 事件或 MouseRightButtonDown 验证了这一点?我本来希望 Source/OriginalSource 始终指向实际引发事件的框架元素。如果您不确定,可以使用 Snoop 或 WPF Inspector 之类的工具来确定事件的来源和处理位置。无论如何,如果您使用 MVVM,为 TreeViewItem 创建 Style 并将其 IsSelected 属性绑定到视图模型中的某些内容可能是一种更简洁的方法。 我知道这是一个迟到的响应,但问题是一样的,所以我不能使用这个解决方案。感谢您抽出宝贵时间回答,【参考方案3】:你可以为它使用一个行为:
using System.Windows;
using System.Windows.Controls;
using System.Windows.Interactivity;
public class SelectOnRMBBehavior : Behavior<FrameworkElement>
public static readonly DependencyProperty CurrentItemProperty = DependencyProperty.Register("CurrentItem", typeof(TreeViewItem), typeof(SelectOnRMBBehavior), new PropertyMetadata(null));
public TreeViewItem CurrentItem
get
return (TreeViewItem)GetValue(CurrentItemProperty);
set
SetValue(CurrentItemProperty, value);
protected override void OnAttached()
base.OnAttached();
AssociatedObject.PreviewMouseRightButtonDown += AssociatedObject_PreviewMouseRightButtonDown;
private void AssociatedObject_PreviewMouseRightButtonDown(object sender, System.Windows.Input.MouseButtonEventArgs e)
if (CurrentItem!=null)
CurrentItem.IsSelected = true;
protected override void OnDetaching()
AssociatedObject.PreviewMouseRightButtonDown -= AssociatedObject_PreviewMouseRightButtonDown;
base.OnDetaching();
然后将你的行为放在将被点击的元素上(我想是 DataTemplate 中的一个容器):
<StackPanel>
<i:Interaction.Behaviors>
<b:SelectOnRMBBehavior CurrentItem="Binding RelativeSource=RelativeSource AncestorType=TreeViewItem"/>
</i:Interaction.Behaviors>
</StackPanel>
【讨论】:
以上是关于调用 WPF 选择 TreeView 项作为上下文菜单的主要内容,如果未能解决你的问题,请参考以下文章
在 XAML 中的 TreeView 中未选择任何内容时禁用 TreeView 上下文菜单项