当数据网格内部的弹出框捕获鼠标时,WPF弹出框不会自动关闭

本文关键字:WPF 网格 数据网 数据 内部 鼠标 | 更新日期: 2023-09-27 17:53:33

我有一个与StaysOpen=False的弹出窗口,所以我想通过点击弹出窗口外的任何地方来关闭它。在弹出框中,我有一个DataGrid。如果我打开弹出窗口,然后点击其他地方的弹出窗口将被关闭。但它不会发生,如果之前点击弹出窗口外,我会点击在DataGrid列标题。测试XAML:

<Window x:Class="Test.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" VerticalAlignment="Center" HorizontalAlignment="Center" Foreground="Black">
<Grid>
    <ToggleButton x:Name="btn" VerticalAlignment="Top">Open</ToggleButton>
    <Popup StaysOpen="False" IsOpen="{Binding IsChecked, ElementName=btn}" > 
        <DataGrid Width="150" Height="150">
            <DataGrid.Columns>
                <DataGridTextColumn Header="Column" />
            </DataGrid.Columns>
        </DataGrid>
    </Popup>
</Grid>
</Window>

我认为它的发生是因为列标题捕获鼠标点击和弹出不再接收鼠标事件。我试图在LostMouseCapture事件上添加处理程序,以便通过弹出窗口捕获鼠标,但它似乎不那么容易工作。什么好主意吗?

当数据网格内部的弹出框捕获鼠标时,WPF弹出框不会自动关闭

也许会有帮助。附加行为:

public class DataGridColumnHeaderReleaseMouseCaptureBehavior {
    public static DataGrid GetReleaseDGCHeaderBehavior(DependencyObject obj) {
        return (DataGrid)obj.GetValue(ReleaseDGCHeaderBehaviorProperty);
    }
    public static void SetReleaseDGCHeaderBehavior(DependencyObject obj, Boolean value) {
        obj.SetValue(ReleaseDGCHeaderBehaviorProperty, value);
    }
    public static readonly DependencyProperty ReleaseDGCHeaderBehaviorProperty =
        DependencyProperty.RegisterAttached("ReleaseDGCHeaderBehavior",
            typeof(DataGrid),
            typeof(DataGridColumnHeaderReleaseMouseCaptureBehavior),
            new UIPropertyMetadata(default(DataGrid), OnReleaseDGCHeaderBehaviorPropertyChanged));
    private static Popup _popup;
    private static void OnReleaseDGCHeaderBehaviorPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) {
        var oldGrid = (DataGrid)e.OldValue;
        if (oldGrid != null)
            oldGrid.MouseLeave -= OnMouseLeave;
        var refSender = d as Popup;
        _popup = refSender;
        if (refSender != null) {
            var refGrid = e.NewValue as DataGrid;
            if (refGrid != null) {
                refGrid.MouseLeave += OnMouseLeave;
            }
        }
    }
    static void OnMouseLeave(object sender, MouseEventArgs args) {
        if (_popup != null)
            typeof(Popup).GetMethod("EstablishPopupCapture", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance).Invoke(_popup, null);
    }
}
XAML:

<Popup x:Name="popup"
bhvrs:DataGridColumnHeaderReleaseMouseCaptureBehavior.ReleaseDGCHeaderBehavior="{Binding ElementName=dataGrid}">
  <DataGrid x:Name="dataGrid"/>
</Popup>

我想你只是偶然发现了一个普通的老bug。我复制了这个,但找不到一个合理的方法让它工作。我认为你应该向微软投诉。这看起来像是一个捕获鼠标的组件,取消捕获它不会将捕获恢复到最初捕获的组件。

我最近有一个类似的问题,虽然不完全相同,它是在Silverlight。我用GetTemplatedParent函数搜索所需的控件(在你的情况下,我猜是弹出式),在"行为不当"控件的所需事件处理程序中,并以编程方式做我想做的事情。

这不是一个好的解决方案,并不能解决所有的问题,但你可以试一试。一定要注释一下你所做的事情,因为这会把事情搞得一团糟。

我遇到了同样的问题,并做了如下操作:

 private void YourDataGrid_PreviewMouseLeftButtonUp(object sender, MouseButtonEventArgs e)
    {
         YourDataGrid.CaptureMouse();
         YourDataGrid.ReleaseMouseCapture();
    }