弹出窗口动画正在运行时,WPF窗口消息未触发

时间:2013-04-07 20:27:25

标签: wpf animation popup

我在WPF中创建了一个小弹出窗口,它显示并隐藏了500毫秒的淡入淡出动画。

弹出TextBox控件的PreviewMouseUp时弹出窗口,并在TextBox焦点丢失时隐藏。

问题在于,如果我有两个这样的TextBox,Popup窗口的动画似乎会阻止在动画进行时发送到主窗口的所有Window Messages。第二个TextBox的PreviewMouseUp仅在第一个TextBox的Popup动画完成后立即触发。

有没有办法让我的弹出窗口的淡入淡出动画在动画运行时不阻止窗口消息?

示例XAML文件:

        <Window x:Class="WpfApplication4.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>
                <TextBox HorizontalAlignment="Left" Height="23" Margin="22,26,0,0" TextWrapping="Wrap" Text="TextBox" VerticalAlignment="Top" Width="120" PreviewMouseUp="TextBox_PreviewMouseUp_1" LostFocus="TextBox_LostFocus_1"/>
                <TextBox HorizontalAlignment="Left" Height="23" Margin="22,54,0,0" TextWrapping="Wrap" Text="TextBox" VerticalAlignment="Top" Width="120" PreviewMouseUp="TextBox_PreviewMouseUp_1" LostFocus="TextBox_LostFocus_1"/>

            </Grid>
        </Window>

示例代码文件:

        using System;
        using System.Collections.Generic;
        using System.Linq;
        using System.Text;
        using System.Threading.Tasks;
        using System.Windows;
        using System.Windows.Controls;
        using System.Windows.Controls.Primitives;
        using System.Windows.Data;
        using System.Windows.Documents;
        using System.Windows.Input;
        using System.Windows.Media;
        using System.Windows.Media.Animation;
        using System.Windows.Media.Imaging;
        using System.Windows.Navigation;
        using System.Windows.Shapes;

        namespace WpfApplication4
        {
            /// <summary>
            /// Interaction logic for MainWindow.xaml
            /// </summary>
            public partial class MainWindow : Window
            {
                public MainWindow()
                {
                    InitializeComponent();
                }

                private void TextBox_PreviewMouseUp_1(object sender, MouseButtonEventArgs e)
                {
                    Popup p = new Popup();
                    p.Width = 100;
                    p.Height = 100;
                    p.Placement = PlacementMode.Left;
                    p.PlacementTarget = (TextBox)sender;
                    p.Child = new Border();
                    p.IsOpen = true;
                    ((TextBox)sender).Tag = p;
                }

                private void TextBox_LostFocus_1(object sender, RoutedEventArgs e)
                {
                    Popup p = (Popup)((TextBox)sender).Tag;
                    DoubleAnimation anim = new DoubleAnimation(100, 0, new Duration(new TimeSpan(0, 0, 1)));
                    p.BeginAnimation(WidthProperty, anim);
                }

            }
        }

如果快速单击两个文本框,您会注意到在动画运行时其他弹出窗口不会出现(并且文本框无法获得焦点)。

到目前为止,我发现,如果动画真的很密集(高帧率),窗口消息会被阻止,直到动画完成。如果我将应用程序帧速率设置为较低的值,如30FPS,则问题就会消失。但这对我来说不是一个选择,因为我不希望动画尽可能顺畅。

1 个答案:

答案 0 :(得分:1)

之所以发生这种情况,是因为我正在设置弹出窗口边缘的动画 - 当窗口大小经常变化时会生成大量窗口消息。 (在动画期间,应用程序的整个windows messagepump中的哪个程序堵塞)。

因此,解决方案不是为弹出窗口本身设置动画,而是设置弹出窗口的子控件(例如面板),并将弹出窗口设置为允许透明度,以便我们获得相同的效果,但实际上不是动画的实际窗口大小