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 吗?我提供的方法的主体是否正确?

请帮忙。谢谢。

4

5 回答 5

7

OnApplyTemplate如果您使用的ControlTemplate是控件,则使用该方法将起作用。例如,如果您已经子类化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)
    {
        //...
    }
}

但是,我认为这种方法不适用于您的情况。您可以使用 Loaded 事件ListBoxItemSlider在事件处理程序的可视化树中找到

<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)
{

}

获取VisualChild

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;
}
于 2011-02-25T00:23:41.010 回答
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 仅与控制模板相关,而与数据模板无关。

于 2011-02-25T15:47:20.977 回答
0

看看这个问题和答案。

我认为使用命令是最好的选择。

编辑 一个很好的教程

于 2011-02-24T23:27:37.060 回答
0

您可以EventSetter在设置模板的样式中使用:

<Style TargetType="{x:Type ListBoxItem}">
      <EventSetter Event="MouseWheel" Handler="GroupListBox_MouseWheel" />
      <Setter Property="Template" ... />
</Style>
于 2012-05-27T16:17:06.477 回答
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>
于 2014-12-22T20:42:25.277 回答