Всплывающее окно wpf не закрывается автоматически, когда сетка данных внутри всплывающего окна захватывает мышь

У меня всплывающее окно с 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 событие, чтобы захватить мышь назад всплывающим окном, но это, кажется, не так просто. Есть идеи?

4 ответа

Может быть, это поможет. Прикрепленное поведение:

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>

Я думаю, что вы наткнулись на просто старую ошибку. Я воспроизвел это и не смог найти разумный способ заставить его работать. Я думаю, вы должны сообщить об ошибке в Microsoft. Похоже, компонент, который захватывает мышь и снимает с нее, не восстанавливает захват в исходный компонент захвата.

Недавно у меня была похожая проблема, хотя и не совсем та же, что и в Silverlight. Я взломал свой путь через него, выполнив поиск необходимого элемента управления (в вашем случае, я полагаю, всплывающего окна) с помощью функции GetTemplatedParent, в требуемом обработчике события элемента управления "неправильно себя ведущий", и программно сделал то, что я хотел с ним сделать.

Это не очень хорошее решение, и оно не решает всех проблем, но вы можете попробовать. Обязательно прокомментируйте, что вы сделали, потому что это может превратиться в беспорядок.

У меня была такая же проблема, и сделал что-то вроде этого:

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

но я ищу что-то лучшее еще...

Другие вопросы по тегам