5

我正在将我的程序从 WinForms 移植到 WPF,并且遇到了一些拖放问题。它应该允许从 TreeView(它就像一个文件浏览器)拖动到打开文件的文本框。但是,WPF 版本的行为类似于TreeViewItem自动复制和粘贴 的标题文本。我想我只是有一些混淆?可能是DataObject东西。

功能齐全的相关 WinForms 代码:

private void treeView1_MouseMove(object sender, MouseEventArgs e)
{
    if (e.Button != MouseButtons.Left) return;
    TreeNode node = treeView1.GetNodeAt(e.Location);
    if (node != null) treeView1.DoDragDrop(node, DragDropEffects.Move);
}

textbox[i].DragDrop += (o, ee) =>
{
     if (ee.Data.GetDataPresent(typeof(TreeNode)))
     {
         TreeNode node = (TreeNode)ee.Data.GetData(typeof(TreeNode));   
         ((Textbox)o).Text = File.ReadAllLines(pathRoot + node.Parent.FullPath);
         ...

应该做同样事情的 WPF 代码:

private void TreeView_PreviewMouseLeftButtonDown(object sender, MouseButtonEventArgs e)
{
    TreeViewItem item = e.Source as TreeViewItem;
    if (item != null)
    {
        DataObject dataObject = new DataObject();
        dataObject.SetData(DataFormats.StringFormat, GetFullPath(item));
        DragDrop.DoDragDrop(item, dataObject, DragDropEffects.Move);
    }
}

//textbox[i].PreviewDrop += textbox_Drop;
private void textbox_Drop(object sender, DragEventArgs e)
{
    TreeViewItem node = (TreeViewItem)e.Data.GetData(typeof(TreeViewItem)); //null?
    ((Textbox)sender).Text = ""; 
    //this is being executed BUT then the node's header text is being pasted
    //also, how do I access the DataObject I passed?
}

问题:在我的 WPF 版本中,我将文本框的文本设置为空(作为测试),这发生了,但随后粘贴了 TreeViewItem 的标题文本,这不是我想要的。

问题:将此 WinForms 代码移植到 WPF 的正确方法是什么?为什么在 WPF 版本中粘贴文本?我该如何防止呢?我使用了正确的事件吗?如何访问DataObjectintextbox_Drop以便可以像在 WinForms 版本中一样打开文件?为什么 WPF 版本中 TreeViewItem 节点总是为空?

4

5 回答 5

4

啊,到底是什么,我将把我的评论扩展到一个答案:

如前所述,要阅读的链接是:http: //msdn.microsoft.com/en-us/library/hh144798.aspx

简而言之,TextBox派生控件已经实现了基本拖放操作的大部分“胆量”,建议您扩展它而不是提供显式DragEnter/DragOver/Drop处理程序。

假设树“数据”结构如下:

public class TreeThing
{
   public string Description { get; set; }
   public string Path { get; set; }
}

处理程序可能看起来像这样:

this.tb.AddHandler(UIElement.DragOverEvent, new DragEventHandler((sender, e) =>
    {
        e.Effects = !e.Data.GetDataPresent("treeThing") ? 
            DragDropEffects.None : 
            DragDropEffects.Copy;                    
    }), true);

this.tb.AddHandler(UIElement.DropEvent, new DragEventHandler((sender, e) =>
{
    if (e.Data.GetDataPresent("treeThing"))
    {
        var item = e.Data.GetData("treeThing") as TreeThing;
        if (item != null)
        {
            tb.Text = item.Path;
            // TODO: Actually open up the file here
        }
    }
}), true);

只是为了咯咯笑,这是一个快速而肮脏的测试应用程序,它使用 Reactive Extensions (Rx) 进行拖动开始的东西是纯粹的炫耀:

XAML:

<Window x:Class="WpfApplication1.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525">
    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition/>
            <ColumnDefinition/>
        </Grid.ColumnDefinitions>
        <TreeView x:Name="tree" Grid.Column="0" ItemsSource="{Binding TreeStuff}" DisplayMemberPath="Description"/>
        <TextBox x:Name="tb" Grid.Column="1" AllowDrop="True" Text="Drop here" Height="30"/>
    </Grid>
</Window>

讨厌的代码隐藏(懒得 MVVM 这个):

using System;
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Reactive.Linq;
using System.Runtime.CompilerServices;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Input;
using System.Windows.Media;

namespace WpfApplication1
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window, INotifyPropertyChanged
    {
        public MainWindow()
        {
            InitializeComponent();
            TreeStuff = new ObservableCollection<TreeThing>()
                {
                    new TreeThing() { Description="file 1",  Path = @"c:\temp\test.txt" },
                    new TreeThing() { Description="file 2", Path = @"c:\temp\test2.txt" },
                    new TreeThing() { Description="file 3", Path = @"c:\temp\test3.txt" },
                };

            var dragStart = 
                from mouseDown in 
                    Observable.FromEventPattern<MouseButtonEventHandler, MouseEventArgs>(
                        h => tree.PreviewMouseDown += h, 
                        h => tree.PreviewMouseDown -= h)
                let startPosition = mouseDown.EventArgs.GetPosition(null)
                from mouseMove in 
                    Observable.FromEventPattern<MouseEventHandler, MouseEventArgs>(
                        h => tree.MouseMove += h, 
                        h => tree.MouseMove -= h)
                let mousePosition = mouseMove.EventArgs.GetPosition(null)
                let dragDiff = startPosition - mousePosition
                where mouseMove.EventArgs.LeftButton == MouseButtonState.Pressed && 
                    (Math.Abs(dragDiff.X) > SystemParameters.MinimumHorizontalDragDistance ||
                    Math.Abs(dragDiff.Y) > SystemParameters.MinimumVerticalDragDistance)
                select mouseMove;

            dragStart.ObserveOnDispatcher().Subscribe(start =>
                {
                    var nodeSource = this.FindAncestor<TreeViewItem>(
                        (DependencyObject)start.EventArgs.OriginalSource);
                    var source = start.Sender as TreeView;
                    if (nodeSource == null || source == null)
                    {
                        return;
                    }
                    var data = (TreeThing)source
                        .ItemContainerGenerator
                        .ItemFromContainer(nodeSource);
                    DragDrop.DoDragDrop(nodeSource, new DataObject("treeThing", data), DragDropEffects.All);
                });

            this.tb.AddHandler(UIElement.DragOverEvent, new DragEventHandler((sender, e) =>
                {
                    e.Effects = !e.Data.GetDataPresent("treeThing") ? 
                        DragDropEffects.None : 
                        DragDropEffects.Copy;                    
                }), true);

            this.tb.AddHandler(UIElement.DropEvent, new DragEventHandler((sender, e) =>
            {
                if (e.Data.GetDataPresent("treeThing"))
                {
                    var item = e.Data.GetData("treeThing") as TreeThing;
                    if (item != null)
                    {
                        tb.Text = item.Path;
                       // TODO: Actually open up the file here
                    }
                }
            }), true);
            this.DataContext = this;
        }


        private T FindAncestor<T>(DependencyObject current)
            where T:DependencyObject
        {
            do
            {
                if (current is T)
                {
                    return (T)current;
                }
                current = VisualTreeHelper.GetParent(current);
            }
            while (current != null);
            return null;
        }

        public event PropertyChangedEventHandler PropertyChanged;

        public ObservableCollection<TreeThing> TreeStuff { get; set; }

        protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
        {
            PropertyChangedEventHandler handler = PropertyChanged;
            if (handler != null)
            {
                handler(this, new PropertyChangedEventArgs(propertyName));
            }
        }
    }

    public class TreeThing
    {
        public string Description { get; set; }
        public string Path { get; set; }
    }
}
于 2013-07-18T17:14:03.600 回答
3

你有不止一个问题,足以让这变得困难。第一个问题是您的拖动对象错误,您正在拖动一个字符串但仍在检查 TreeViewItem。只需使用与 Winforms 中相同的方法,拖动节点即可。第二个问题是 TextBox 已经实现了 D+D 支持,这妨碍了您的代码。以及你看到文字的原因是在下降后出现的。

让我们首先解决阻力的开始。您需要做一些额外的工作,因为您开始拖动的方式会干扰 TreeView 的正常使用,因此很难选择一个节点。仅当鼠标移动足够远时才开始拖动:

    private Point MouseDownPos;

    private void treeView1_PreviewMouseDown(object sender, MouseButtonEventArgs e) {
        MouseDownPos = e.GetPosition(treeView1);
    }

    private void treeView1_PreviewMouseMove(object sender, MouseEventArgs e) {
        if (e.LeftButton == MouseButtonState.Released) return;
        var pos = e.GetPosition(treeView1);
        if (Math.Abs(pos.X - MouseDownPos.X) >= SystemParameters.MinimumHorizontalDragDistance ||
            Math.Abs(pos.Y - MouseDownPos.Y) >= SystemParameters.MinimumVerticalDragDistance) {
            TreeViewItem item = e.Source as TreeViewItem;
            if (item != null) DragDrop.DoDragDrop(item, item, DragDropEffects.Copy);
        }
    }

现在是drop,您需要实现DragEnter、DragOver 和Drop 事件处理程序,以避免TextBox 内置的默认D+D 支持妨碍您。将 e.Handled 属性设置为 true 是必要的:

    private void textBox1_PreviewDragEnter(object sender, DragEventArgs e) {
        if (e.Data.GetDataPresent(typeof(TreeViewItem))) e.Effects = e.AllowedEffects;
        e.Handled = true;
    }

    private void textBox1_PreviewDrop(object sender, DragEventArgs e) {
        var item = (TreeViewItem)e.Data.GetData(typeof(TreeViewItem));
        textBox1.Text = item.Header.ToString();   // Replace this with your own code
        e.Handled = true;
    }

    private void textBox1_PreviewDragOver(object sender, DragEventArgs e) {
        e.Handled = true;
    }
于 2013-07-18T17:45:42.100 回答
1

问题:在我的 WPF 版本中,我将文本框的文本设置为空(作为测试),这会发生,但随后粘贴了 TreeViewItem 的标题文本,这不是我想要的。

我认为父 UI 元素正在处理(并因此覆盖)该Drop事件,因此您不会获得预期的结果。事实上,在尝试重新创建您的问题时,我什至无法触发 TextBox.Drop 事件。但是,使用 TextBox 的 PreviewDrop 事件,我能够得到(我认为)您的预期结果。尝试这个:

    private void textBox1_PreviewDrop(object sender, DragEventArgs e)
    {
        TextBox tb = sender as TextBox;
        if (tb != null)
        {
            // If the DataObject contains string data, extract it.
            if (e.Data.GetDataPresent(DataFormats.StringFormat))
            {
                string fileName = e.Data.GetData(DataFormats.StringFormat) as string;
                using (StreamReader s = File.OpenText(fileName))
                {
                    ((TextBox)sender).Text = s.ReadToEnd();
                }
            }
        }
        e.Handled = true; //be sure to set this to true
    }

我认为代码片段应该回答你提出的大部分问题,除了这个:

为什么 WPF 版本中 TreeViewItem 节点总是为空?

您在 DragDrop 事件中传递的DataObject不支持传递TreeViewItem. 在您的代码(和我的)中,我们指定数据格式DataFormats.StringFormat不能转换为TreeViewItem.

于 2013-07-18T14:22:11.033 回答
0

GetFullPath似乎输出了错误的值。您要拖放的是Header,您可以直接从item. 还要记住,下面的方法MouseMove EventTreeView.

private void TreeView_MouseMove(object sender, MouseButtonEventArgs e)
{
    if (e.LeftButton != MouseButtonState.Pressed) return;
    TreeViewItem item = e.Source as TreeViewItem;
    if (item != null)
    {
        DataObject dataObject = new DataObject();
        dataObject.SetData(DataFormats.StringFormat, item.Header);
        DragDrop.DoDragDrop(item, dataObject, DragDropEffects.Move);
    }
}

我确实基于文本而不是TreeViewItem( e.Data.GetData(typeof(string)).ToString()) 创建了放置部分,但最令人惊讶的是它甚至不是必需的。如果您打开一个新的 C# WPF 项目,在其上放置 aTreeView和 a TextBox(更新 XAML 部分)并复制上面的代码,您可以将文本从其中拖放TreeView到其中,TextBox而无需执行任何其他操作!文本被复制到TextBox不考虑的Drop handling.

于 2013-07-18T15:26:50.387 回答
-1

我使用的是正确的事件吗?:我认为您使用的是正确的事件,但我认为您的代码中有几个问题。我假设您已将树视图的 DataContext 设置为真实项目并使用绑定。

  1. 如何访问 textbox_Drop 中的 DataObject ?--> 要获取 DataObject,您必须通过递归获取真实项目(可能的其他解决方案)

    DependencyObject k = VisualTreeHelper.HitTest(tv_treeView, DagEventArgs.GetPosition(lv_treeView)).VisualHit;
    
    while (k != null)
        {
            if (k is TreeViewItem)
            {
                TreeViewItem treeNode = k as TreeViewItem;
    
                // Check if the context is your desired type
                if (treeNode.DataContext is YourType)
                {
                    // save the item
                    targetTreeViewItem = treeNode;
    
                    return;
                }
            }
            else if (k == tv_treeview)
            {
                Console.WriteLine("Found treeview instance");
                return;
            }
    
            // Get the parent item if no item from YourType was found
            k = VisualTreeHelper.GetParent(k);
        }
    
  2. 为什么在 WPF 版本中粘贴文本?--> 显示 Header 是因为(我假设)它就像您的项目上的 tostring 方法。如果未指定复杂项目的绑定,则执行 ToString 方法。尽量不要在 drop 事件的处理程序中直接设置 Text。将数据上下文设置为您的项目(到您在 1. 中找到的项目),然后通过 XAML 指定绑定路径。(用于显示)

于 2013-07-16T05:32:28.573 回答