防止 TreeView 中的自动水平滚动不起作用。 MVVM,XAML 方法
Prevent Automatic Horizontal Scroll in TreeView is not working. MVVM, XAML approach
每当在我的树视图中选择一个节点时,它会自动水平滚动到该项目。 I've found the way to disable this。如果我在后面的代码中使用这段代码,它会完美地工作:
<TreeView>
<TreeView.ItemContainerStyle>
<Style TargetType="TreeViewItem">
<EventSetter Event="RequestBringIntoView" Handler="TreeViewItem_RequestBringIntoView"/>
</Style>
</TreeView.ItemContainerStyle>
</TreeView>
private void TreeViewItem_RequestBringIntoView(object sender, RequestBringIntoViewEventArgs e)
{
e.Handled = true;
}
但是,如果我使用 MVVM,我无法禁用项目的水平滚动:
我的window:
<Window x:Class="TreeViewWpfApplication.MainWindow"
. . . . .
xmlns:i="clr-namespace:System.Windows.Interactivity;assembly=System.Windows.Interactivity"
xmlns:ei=http://schemas.microsoft.com/expression/2010/interactions>
<TreeView Grid.Column="1" Margin="5" Background="Green">
<i:Interaction.Triggers>
<i:EventTrigger EventName="RequestBringIntoView">
<ei:CallMethodAction MethodName="RequestBringIntoView_Handler" TargetObject="{Binding}"/>
</i:EventTrigger>
</i:Interaction.Triggers>
<TreeView>
<TreeViewItem Header="---Level 1" >
<TreeViewItem Header="--- Level 2.1" >
<TreeViewItem Header="--- Level 3.1" >
</TreeViewItem>
</TreeViewItem>
</TreeViewItem>
<TreeViewItem Header="Level 2.3" />
</TreeView>
</Window>
视图模型:
public void RequestBringIntoView_Handler(object sender, RequestBringIntoViewEventArgs e)
{
e.Handled = true;
}
为什么我不能通过 MVVM 方法停止自动水平滚动到项目?
我相信您总是可以循环遍历 TreeView
的所有 TreeViewItem
,并在附加每个 TreeView
之前从附加在 TreeView
上的 Interaction.Triggers
克隆每个 TriggerBase
克隆一个到每个 TreeViewItem
的 Interaction.Triggers
。
我对微软很失望,自从我开始学习如何编程以来,这个名字让我感到骄傲,多年来一直是我无尽的灵感。但坦率地说,微软有很多事情让我们感到失望。您的代码实际上应该可以正常工作。为什么?我试过了,事件 RequestBringIntoView
实际上从 TreeViewItem
上升到 TreeView
。事实上,当您直接在 TreeView
上添加事件处理程序时,您会看到事件处理程序已正常触发。但是使用 Interaction
的设置处理程序的非常等效的形式不能那样工作。太可怕了。很明显,它旨在以 MVVM 方式设置事件处理程序,但它非常有限。
我不得不做一个解决方法,我们使用自定义附加 属性 来允许在样式中设置 Interaction.Triggers
。但是我不得不说它不是很漂亮。您需要显式声明 Array
of TriggerBase
(我以前做过类似的事情,但从未找到更好的解决方案)。接下来,您需要使用代理为 EventTrigger
绑定 TargetObject
(因为我们将触发器放在一个数组中,并且它与可视化树分离)。
这是附加的自定义代码属性:
//add some using alias like this first
//using i = System.Windows.Interactivity;
public static class InteractionX
{
public static readonly DependencyProperty TriggersProperty
= DependencyProperty.RegisterAttached("Triggers", typeof(i.TriggerBase[]),
typeof(InteractionX), new PropertyMetadata(triggersChanged));
public static i.TriggerBase[] GetTriggers(DependencyObject o){
return o.GetValue(TriggersProperty) as i.TriggerBase[];
}
public static void SetTriggers(DependencyObject o, i.TriggerBase[] value)
{
o.SetValue(TriggersProperty, value);
}
static void triggersChanged(DependencyObject o, DependencyPropertyChangedEventArgs e)
{
var triggers = e.NewValue as i.TriggerBase[];
var currentTriggers = i.Interaction.GetTriggers(o);
currentTriggers.Clear();
foreach (var t in triggers)
{
t.Detach();
currentTriggers.Add(t);
}
}
}
这里是 XAML:
<TreeView>
<TreeView.Resources>
<DiscreteObjectKeyFrame x:Key="proxy" Value="{Binding}"/>
</TreeView.Resources>
<TreeView.ItemContainerStyle>
<Style TargetType="TreeViewItem">
<Setter Property="local:InteractionX.Triggers">
<Setter.Value>
<x:Array Type="{x:Type i:TriggerBase}">
<i:EventTrigger EventName="RequestBringIntoView">
<ei:CallMethodAction MethodName="bringIntoViewHandler"
TargetObject="{Binding Value, Source={StaticResource proxy}}"/>
</i:EventTrigger>
</x:Array>
</Setter.Value>
</Setter>
</Style>
</TreeView.ItemContainerStyle>
</TreeView>
看来 TreeViewItem
上设置的 Interaction.Triggers
可以处理来自后代 TreeViewItems 的冒泡 RequestBringIntoView
但正如我所说,将其设置在 [=13= 上很遗憾] 不起作用。
每当在我的树视图中选择一个节点时,它会自动水平滚动到该项目。 I've found the way to disable this。如果我在后面的代码中使用这段代码,它会完美地工作:
<TreeView>
<TreeView.ItemContainerStyle>
<Style TargetType="TreeViewItem">
<EventSetter Event="RequestBringIntoView" Handler="TreeViewItem_RequestBringIntoView"/>
</Style>
</TreeView.ItemContainerStyle>
</TreeView>
private void TreeViewItem_RequestBringIntoView(object sender, RequestBringIntoViewEventArgs e)
{
e.Handled = true;
}
但是,如果我使用 MVVM,我无法禁用项目的水平滚动:
我的window:
<Window x:Class="TreeViewWpfApplication.MainWindow"
. . . . .
xmlns:i="clr-namespace:System.Windows.Interactivity;assembly=System.Windows.Interactivity"
xmlns:ei=http://schemas.microsoft.com/expression/2010/interactions>
<TreeView Grid.Column="1" Margin="5" Background="Green">
<i:Interaction.Triggers>
<i:EventTrigger EventName="RequestBringIntoView">
<ei:CallMethodAction MethodName="RequestBringIntoView_Handler" TargetObject="{Binding}"/>
</i:EventTrigger>
</i:Interaction.Triggers>
<TreeView>
<TreeViewItem Header="---Level 1" >
<TreeViewItem Header="--- Level 2.1" >
<TreeViewItem Header="--- Level 3.1" >
</TreeViewItem>
</TreeViewItem>
</TreeViewItem>
<TreeViewItem Header="Level 2.3" />
</TreeView>
</Window>
视图模型:
public void RequestBringIntoView_Handler(object sender, RequestBringIntoViewEventArgs e)
{
e.Handled = true;
}
为什么我不能通过 MVVM 方法停止自动水平滚动到项目?
我相信您总是可以循环遍历 TreeView
的所有 TreeViewItem
,并在附加每个 TreeView
之前从附加在 TreeView
上的 Interaction.Triggers
克隆每个 TriggerBase
克隆一个到每个 TreeViewItem
的 Interaction.Triggers
。
我对微软很失望,自从我开始学习如何编程以来,这个名字让我感到骄傲,多年来一直是我无尽的灵感。但坦率地说,微软有很多事情让我们感到失望。您的代码实际上应该可以正常工作。为什么?我试过了,事件 RequestBringIntoView
实际上从 TreeViewItem
上升到 TreeView
。事实上,当您直接在 TreeView
上添加事件处理程序时,您会看到事件处理程序已正常触发。但是使用 Interaction
的设置处理程序的非常等效的形式不能那样工作。太可怕了。很明显,它旨在以 MVVM 方式设置事件处理程序,但它非常有限。
我不得不做一个解决方法,我们使用自定义附加 属性 来允许在样式中设置 Interaction.Triggers
。但是我不得不说它不是很漂亮。您需要显式声明 Array
of TriggerBase
(我以前做过类似的事情,但从未找到更好的解决方案)。接下来,您需要使用代理为 EventTrigger
绑定 TargetObject
(因为我们将触发器放在一个数组中,并且它与可视化树分离)。
这是附加的自定义代码属性:
//add some using alias like this first
//using i = System.Windows.Interactivity;
public static class InteractionX
{
public static readonly DependencyProperty TriggersProperty
= DependencyProperty.RegisterAttached("Triggers", typeof(i.TriggerBase[]),
typeof(InteractionX), new PropertyMetadata(triggersChanged));
public static i.TriggerBase[] GetTriggers(DependencyObject o){
return o.GetValue(TriggersProperty) as i.TriggerBase[];
}
public static void SetTriggers(DependencyObject o, i.TriggerBase[] value)
{
o.SetValue(TriggersProperty, value);
}
static void triggersChanged(DependencyObject o, DependencyPropertyChangedEventArgs e)
{
var triggers = e.NewValue as i.TriggerBase[];
var currentTriggers = i.Interaction.GetTriggers(o);
currentTriggers.Clear();
foreach (var t in triggers)
{
t.Detach();
currentTriggers.Add(t);
}
}
}
这里是 XAML:
<TreeView>
<TreeView.Resources>
<DiscreteObjectKeyFrame x:Key="proxy" Value="{Binding}"/>
</TreeView.Resources>
<TreeView.ItemContainerStyle>
<Style TargetType="TreeViewItem">
<Setter Property="local:InteractionX.Triggers">
<Setter.Value>
<x:Array Type="{x:Type i:TriggerBase}">
<i:EventTrigger EventName="RequestBringIntoView">
<ei:CallMethodAction MethodName="bringIntoViewHandler"
TargetObject="{Binding Value, Source={StaticResource proxy}}"/>
</i:EventTrigger>
</x:Array>
</Setter.Value>
</Setter>
</Style>
</TreeView.ItemContainerStyle>
</TreeView>
看来 TreeViewItem
上设置的 Interaction.Triggers
可以处理来自后代 TreeViewItems 的冒泡 RequestBringIntoView
但正如我所说,将其设置在 [=13= 上很遗憾] 不起作用。