我可以通过DataTrigger(XAML)调用函数吗?

时间:2014-09-25 11:18:14

标签: c# wpf events command datatrigger

我在.xaml.cs中有一个函数,我想调用它:

.xaml.cs

private void treeView_SelectedItemChanged(object sender, EventArgs e)
{
        //treeView.ScrollToCenterOfView(sender as TreeViewItem);
}

喜欢这篇文章: Make ListView.ScrollIntoView Scroll the Item into the Center of the ListView (C#) 答案1

但我不使用SelectedItem,我的方法就像:

XAML:

<Style.Triggers>
     <DataTrigger Binding="{Binding Path=IsActive}" Value="true">
          // ###call this function
     </DataTrigger>
</Style.Triggers>

我不知道如何通过更改DataModel来提升此功能。 在每个教程中,我都会通过事件/命令来完成它。

THX

4 个答案:

答案 0 :(得分:1)

只需为这些情况添加另一个稍微有点怪异的选项。

并不是触发器,但是您可以利用通用的FrameworkElement.Tag属性为您要观看的属性设置绑定,然后在更改时运行回调。

例如:

<DataTemplate>
    <StackPanel Tag="{Binding IsActive, NotifyOnTargetUpdated=True}"
                TargetUpdated="CodeBehindMethodToCall">

        <!-- snip rest of template --> 

    </StackPanel>
<DataTemplate>

答案 1 :(得分:0)

我想你不能。 DataTrigger仅用于更改WPF级别的表示。标准方法是使用控件提供的事件作为事件。

答案 2 :(得分:0)

TreeViewItem类有一个IsSelected,重要的是一个Selected事件,所以理论上你可以在创建TreeViewItem时为Selected事件设置和事件处理程序。

希望这一点显而易见,这将无法在MVVM场景中工作,因为您无法访问创建的TreeViewItem实例,在这种情况下,我希望通过TreeView控件的Behavior来执行此操作。 / p>

答案 3 :(得分:0)

这是我使用DataTrigger的Auto-ScrollItemToCenter TreeView的解决方案,因此它不是滚动的所选项目。

在XAML中,我通过DataTrigger添加了一个自定义行为

                <Style TargetType="TreeViewItem" BasedOn="{StaticResource {x:Type TreeViewItem}}">
                    <Style.Triggers>
                        <DataTrigger Binding="{Binding Path=IsSelectedInGrid}" Value="true">
                            <Setter Property="local:TreeViewScrollBehavior.ScrollItemToCenter" Value="true" />
                        </DataTrigger>
                    </Style.Triggers>

我试图在http://www.shujaat.net/2010/08/attached-behaviors.html

这个简单的例子之后创建我自己的行为
public static class TreeViewScrollBehavior
{
    public static DependencyProperty ScrollItemToCenterProperty =
DependencyProperty.RegisterAttached("ScrollItemToCenter", typeof(bool),
typeof(TreeViewScrollBehavior), new UIPropertyMetadata(false, OnScrollItemToCenter));


    public static bool GetScrollItemToCenter(DependencyObject obj)
    {
        return (bool)obj.GetValue(ScrollItemToCenterProperty);
    }

    public static void SetScrollItemToCenter(DependencyObject obj, bool value)
    {
        obj.SetValue(ScrollItemToCenterProperty, value);
    }

    public static void OnScrollItemToCenter(DependencyObject d,
        DependencyPropertyChangedEventArgs e)
    {

        if ((bool)e.NewValue)
        {
            //TreeViewItem t = GetTreeViewItem(d);
            TreeViewItem t = (TreeViewItem)d;
            TryScrollToCenterOfView(GetTree(t), t);

        }
    }

并添加了上述链接

的略微更改的功能
    private static TreeView GetTree(TreeViewItem item)
    {
        var parent = VisualTreeHelper.GetParent(item);
        while (!(parent is TreeView))
        {
            parent = VisualTreeHelper.GetParent(parent);
        }
        return parent as TreeView;
    }


    private static void TryScrollToCenterOfView(ItemsControl itemsControl, UIElement container)
    {
        // Find the container
        //var container =  itemsControl.ItemContainerGenerator.ContainerFromItem(item) as UIElement;
        if (container == null) return;

        // Find the ScrollContentPresenter
        ScrollContentPresenter presenter = null;
        for (Visual vis = container; vis != null && vis != itemsControl; vis = VisualTreeHelper.GetParent(vis) as Visual)
            if ((presenter = vis as ScrollContentPresenter) != null)
                break;
        if (presenter == null) return;

        // Find the IScrollInfo
        var scrollInfo =
            !presenter.CanContentScroll ? presenter :
            presenter.Content as IScrollInfo ??
            FirstVisualChild(presenter.Content as ItemsPresenter) as IScrollInfo ??
            presenter;

        // Compute the center point of the container relative to the scrollInfo
        Size size = container.RenderSize;
        Point center = container.TransformToAncestor((Visual)scrollInfo).Transform(new Point(size.Width / 2, size.Height / 2));
        center.Y += scrollInfo.VerticalOffset;
        center.X += scrollInfo.HorizontalOffset;

        // Adjust for logical scrolling
        if (scrollInfo is StackPanel || scrollInfo is VirtualizingStackPanel)
        {
            double logicalCenter = itemsControl.ItemContainerGenerator.IndexFromContainer(container) + 0.5;
            Orientation orientation = scrollInfo is StackPanel ? ((StackPanel)scrollInfo).Orientation : ((VirtualizingStackPanel)scrollInfo).Orientation;
            if (orientation == Orientation.Horizontal)
                center.X = logicalCenter;
            else
                center.Y = logicalCenter;
        }

        // Scroll the center of the container to the center of the viewport
        if (scrollInfo.CanVerticallyScroll) scrollInfo.SetVerticalOffset(CenteringOffset(center.Y, scrollInfo.ViewportHeight, scrollInfo.ExtentHeight));
        if (scrollInfo.CanHorizontallyScroll) scrollInfo.SetHorizontalOffset(CenteringOffset(center.X, scrollInfo.ViewportWidth, scrollInfo.ExtentWidth));
    }

    private static double CenteringOffset(double center, double viewport, double extent)
    {
        return Math.Min(extent - viewport, Math.Max(0, center - viewport / 2));
    }
    private static DependencyObject FirstVisualChild(Visual visual)
    {
        if (visual == null) return null;
        if (VisualTreeHelper.GetChildrenCount(visual) == 0) return null;
        return VisualTreeHelper.GetChild(visual, 0);
    }

这是我在xaml.cs或DataObject中滚动没有Code的TreeView的完整解决方案。

对用户AwkwardCoder的行为理念的Thx

如果有人需要改进,我希望听到它; - )