在WPF中,如何为在模板中设计的FrameworkElement添加一个EventHandler?

4

我已经在外部资源字典中定义了以下DataTemplate,用于ListBox项目:

<DataTemplate x:Key="MyListBoxItemTemplate" DataType="{x:Type entities:Track}">
    <StackPanel>       
        <TextBlock Text="Here's the slider:" />
        <Slider Name="MySlider" Height="23" Minimum="0" />
    </StackPanel>
</DataTemplate>

我需要为Slider的ValueChanged事件提供一个事件处理程序方法。我不知道应该在哪里编写这段代码,因为在模板中指定控件的事件处理程序是不实际的。

我一直在搜索解决方案,并发现我应该在OnApplyTemplate()方法的覆盖中添加事件处理程序。我的猜测是它应该看起来像这样或类似:

public override void OnApplyTemplate()
{
    base.OnApplyTemplate();
    // Is the following initialization even going to work!?!?
    Slider MySlider = this.FindName("MySlider") as Slider;
    SeekSlider.ValueChanged += 
        new RoutedPropertyChangedEventHandler<double>(SeekSlider_ValueChanged);
}

但是我应该在哪里编写这个方法?OnApplyTemplate重写只适用于ControlTemplates还是我的情况也包括在内?我应该提供ControlTemplate而不是DataTemplate吗?我提供的方法体是否正确?

请帮忙,谢谢。

5个回答

7

如果您正在处理控件的ControlTemplate,那么使用OnApplyTemplate方法是可行的。例如,如果您已经子类化了TextBox,您可以这样做:

public class MyTextBox : TextBox
{
    public override void OnApplyTemplate()
    {
        MySlider MySlider = GetTemplateChild("MySlider") as MySlider;
        if (MySlider != null)
        {
            MySlider.ValueChanged += new RoutedPropertyChangedEventHandler<double>(MySlider_ValueChanged);
        }
        base.OnApplyTemplate();
    }
    void MySlider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
    {
        //...
    }
}

然而,我认为这种方法在你的情况下不起作用。您可以使用ListBoxItem的Loaded事件,并在事件处理程序中在可视树中查找Slider

<ListBox ...>
    <ListBox.ItemContainerStyle>
        <Style TargetType="ListBoxItem">
            <EventSetter Event="Loaded" Handler="ListBoxItem_Loaded"/>
        </Style>
    </ListBox.ItemContainerStyle>
    <!--...-->
</ListBox>

后台代码

private void ListBoxItem_Loaded(object sender, RoutedEventArgs e)
{
    ListBoxItem listBoxItem = sender as ListBoxItem;
    Slider MySlider = GetVisualChild<Slider>(listBoxItem);
    MySlider.ValueChanged += new RoutedPropertyChangedEventHandler<double>(MySlider_ValueChanged);
}
void MySlider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
{

}

GetVisualChild

private static T GetVisualChild<T>(DependencyObject parent) where T : Visual
{
    T child = default(T);

    int numVisuals = VisualTreeHelper.GetChildrenCount(parent);
    for (int i = 0; i < numVisuals; i++)
    {
        Visual v = (Visual)VisualTreeHelper.GetChild(parent, i);
        child = v as T;
        if (child == null)
        {
            child = GetVisualChild<T>(v);
        }
        if (child != null)
        {
            break;
        }
    }
    return child;
}

Loaded事件并不总是有用的,例如当Visibility被设置为Collapsed并稍后再设置为Visible时。在这种情况下,Loaded事件会被触发,但模板尚未应用。一旦控件可见并应用了模板,Loaded事件就不会再次触发。 - Peter Huber

3

很少人知道的是ResourceDictionaries也可以包含CodeBehind。

一般来说,我认为将DataTemplates放在ResourceDictionaries中并不是一个好主意(你的问题就是其中之一的原因),以下是解决方法:

XAML:

<ResourceDictionary 
    x:Class="WpfApplication24.Dictionary1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
                    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
    <DataTemplate x:Key="MyDataTemplate">
        <StackPanel>
        <TextBlock Text="Hello" />
            <Slider ValueChanged="ValueChanged"/>
        </StackPanel>
    </DataTemplate>
</ResourceDictionary>

以及代码后端:

namespace WpfApplication24
{
    public partial class Dictionary1 : ResourceDictionary
    {

        public void ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
        {
            Debug.Write("Hello");
        }

    }
}

无论如何,就像上面Meleak所说的 - OnApplyTemplate仅适用于控件模板而不是数据模板。


谢谢Elad。我明白你想表达的观点。现在一切都变得更加清晰,我想我会开始开发一个ControlTemplate :) - Boris
我稍微思考了一下,我不喜欢这个。在ResourceDictionary上使用ValueChanged方法是不合适的。知道有资源字典的代码后台很好,但这让人感到不舒服。 - Emond
我非常同意。 我甚至更进一步地说,Datatemplate根本不属于资源字典中。 - Elad Katz

0

方法1:使用继承于Slider的自定义控件:

public class SpecialSlider : Slider
{
    public SpecialSlider()
    {
        ValueChanged += OnValueChanged;
    }

    private void OnValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
    {
        // ...
    }
}  

方法2:使用来自System.Windows.Interactivity.dll程序集的行为(可通过NuGet获取):

public class SpecialSliderBehavior : Behavior<Slider>
{
    protected override void OnAttached()
    {
        base.OnAttached();
        AssociatedObject.ValueChanged += OnValueChanged;
    }

    protected override void OnDetaching()
    {
        base.OnDetaching();
        AssociatedObject.ValueChanged -= OnValueChanged;
    }

    private void OnValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
    {
        // ...
    }
}

以下是如何附加它的方法:

...
xmlns:i="http://schemas.microsoft.com/expression/2010/interactivity"
...
<DataTemplate x:Key="MyListBoxItemTemplate" DataType="{x:Type entities:Track}">
    <Slider Name="MySlider">
        <i:Interaction.Behaviors>
            <SpecialSliderBehavior />
        </i:Interaction.Behaviors>
    </Slider>
</DataTemplate>

0

在设置模板时,您可以使用EventSetter

<Style TargetType="{x:Type ListBoxItem}">
      <EventSetter Event="MouseWheel" Handler="GroupListBox_MouseWheel" />
      <Setter Property="Template" ... />
</Style>

0

Erno,谢谢你的回答。我会等待一个更容易理解的解决方案。我以前从未使用过命令,也不理解所提供的解释。 - Boris
我添加了一个很好的教程,它解释并展示了标准命令和自定义命令。 - Emond
谢谢Erno,我浏览了一下(因为我现在太忙了),它看起来很合理。感谢您帮助我理解WPF中的命令概念。 - Boris

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接