我在WPF上需要在闲置事件上反复播放视频,然后恢复到检测用户活动的上一个窗口。
为此,我遵循了回答的https://stackoverflow.com/a/4970019/6696609 Martin Buberl。
在XAML中,我有Videogrid和Window Grid网格,在发现空闲事件的情况下,我更改了彼此的可见性。
这是描述问题说明的源代码https://github.com/davidsilwal/wpfvvideoissue闲置事件应在5秒钟发生。当您单击按钮时,闲置事件发生在10秒
除了
外,它的工作正常对于单击按钮的第一个空闲事件,Videogrid尝试可见,但看不到(只是眨眼(,然后每个下一个空闲事件都很好。
的反馈和建议摆脱眨眼问题的建议是赞赏的。
// set UI on inactivity
private void OnInactivity(object sender, EventArgs e)
{
_inactiveMousePosition = Mouse.GetPosition(this);
System.Windows.Application.Current.Dispatcher.BeginInvoke(new Action(() =>
{
if (WindowGrid.Visibility == Visibility.Visible)
{
WindowGrid.Visibility = Visibility.Collapsed;
}
if (VideoGrid.Visibility == Visibility.Collapsed)
{
WindowState = WindowState.Maximized;
WindowStyle = WindowStyle.None;
//play video
videoplayer.Play();
VideoGrid.Visibility = Visibility.Visible;
}
}));
}
private void OnActivity(object sender, PreProcessInputEventArgs e)
{
var inputEventArgs = e.StagingItem.Input;
if (inputEventArgs is System.Windows.Input.MouseEventArgs || inputEventArgs is KeyboardEventArgs)
{
if (e.StagingItem.Input is System.Windows.Input.MouseEventArgs)
{
var mouseEventArgs = (System.Windows.Input.MouseEventArgs)e.StagingItem.Input;
// no button is pressed and the position is still the same as the application became inactive
if (!(
mouseEventArgs.LeftButton == MouseButtonState.Pressed ||
mouseEventArgs.RightButton == MouseButtonState.Pressed ||
mouseEventArgs.MiddleButton == MouseButtonState.Pressed ||
mouseEventArgs.XButton1 == MouseButtonState.Pressed ||
mouseEventArgs.XButton2 == MouseButtonState.Pressed
//|| _inactiveMousePosition != mouseEventArgs.GetPosition(this)
))
{
return;
}
}
// set UI on activity
System.Windows.Application.Current.Dispatcher.BeginInvoke(new Action(() =>
{
if (VideoGrid.Visibility == Visibility.Visible)
{
WindowState = WindowState.Normal;
WindowStyle = WindowStyle.SingleBorderWindow;
//stop video
videoplayer.Stop();
VideoGrid.Visibility = Visibility.Collapsed;
}
if (WindowGrid.Visibility == Visibility.Collapsed)
{
WindowGrid.Visibility = Visibility.Visible;
}
}));
_activityTimer.Stop();
_activityTimer.Start();
}
}
通过调试您的代码,我可以给您一个答案,但也许不是一个非常令人满意的。
我在您的代码中添加了一些调试输出...
private async void OnInactivity(object sender, EventArgs e)
{
// remember mouse position
_inactiveMousePosition = Mouse.GetPosition(this);
// set UI on inactivity
Debug.WriteLine ( DateTime.Now.ToString("HH:mm:ss.fffffff") + " Inactivity" ) ;
await System.Windows.Application.Current.Dispatcher.InvokeAsync(new Action(() =>
{
Grid1.Visibility = Visibility.Collapsed;
videoplayer.Play();
Grid2.Visibility = Visibility.Visible;
}));
}
and Activitive ...
private async void OnActivity(object sender, PreProcessInputEventArgs e)
{
var inputEventArgs = e.StagingItem.Input;
if (inputEventArgs is System.Windows.Input.MouseEventArgs || inputEventArgs is KeyboardEventArgs)
{
if (e.StagingItem.Input is System.Windows.Input.MouseEventArgs)
{
var mouseEventArgs = (System.Windows.Input.MouseEventArgs)e.StagingItem.Input;
if (!(
mouseEventArgs.LeftButton == MouseButtonState.Pressed ||
mouseEventArgs.RightButton == MouseButtonState.Pressed ||
mouseEventArgs.MiddleButton == MouseButtonState.Pressed ||
mouseEventArgs.XButton1 == MouseButtonState.Pressed ||
mouseEventArgs.XButton2 == MouseButtonState.Pressed
|| _inactiveMousePosition != mouseEventArgs.GetPosition(this)
))
{
return;
}
}
Debug.WriteLine ( DateTime.Now.ToString("HH:mm:ss.fffffff") + " Activity " + inputEventArgs.RoutedEvent.ToString() ) ;
// set UI on activity
await System.Windows.Application.Current.Dispatcher.InvokeAsync(new Action(() =>
{
videoplayer.Stop();
Grid1.Visibility = Visibility.Visible;
Grid2.Visibility = Visibility.Hidden;
}));
_activityTimer.Stop();
_activityTimer.Start();
}
}
按下按钮后,这是输出的一点:
23:45:31.0583181 Activity Mouse.MouseMove
23:45:31.0643177 Activity Mouse.QueryCursor
23:45:31.0653178 Activity Mouse.PreviewMouseMove
23:45:31.0663188 Activity Mouse.MouseMove
23:45:31.0723180 Activity Mouse.QueryCursor
23:45:31.0753176 Activity Mouse.PreviewMouseMove
23:45:31.0763193 Activity Mouse.MouseMove
23:45:36.0722537 Inactivity
23:45:36.0752537 Activity Keyboard.PreviewGotKeyboardFocus
23:45:36.0782576 Activity Keyboard.PreviewKeyboardInputProviderAcquireFocus
23:45:36.0802536 Activity Keyboard.KeyboardInputProviderAcquireFocus
23:45:36.0832529 Activity Keyboard.LostKeyboardFocus
23:45:36.0842549 Activity Keyboard.GotKeyboardFocus
23:45:41.1041883 Inactivity
23:45:42.4981698 Activity Mouse.QueryCursor
23:45:42.5001740 Activity Mouse.PreviewMouseMove
23:45:42.5031736 Activity Mouse.MouseMove
23:45:42.5071741 Activity Mouse.QueryCursor
23:45:42.5091736 Activity Mouse.PreviewMouseMove
23:45:42.5111735 Activity Mouse.MouseMove
不活动5秒后,它试图启动视频,但立即引起与不断变化的焦点有关的一些事件。
我猜这意味着单击按钮可以将按钮成为焦点。启动视频会导致其失去焦点,产生一些事件。
我能够通过过滤这些事件来解决问题。这是我的修改版本的OnActivity函数,在其中我按名称过滤这些事件:
private async void OnActivity(object sender, PreProcessInputEventArgs e)
{
var inputEventArgs = e.StagingItem.Input;
switch ( inputEventArgs.RoutedEvent.Name )
{
case "PreviewGotKeyboardFocus":
case "PreviewKeyboardInputProviderAcquireFocus":
case "KeyboardInputProviderAcquireFocus":
case "LostKeyboardFocus":
case "GotKeyboardFocus":
return ;
}
if (inputEventArgs is System.Windows.Input.MouseEventArgs || inputEventArgs is KeyboardEventArgs)
{
if (e.StagingItem.Input is System.Windows.Input.MouseEventArgs)
{
var mouseEventArgs = (System.Windows.Input.MouseEventArgs)e.StagingItem.Input;
if (!(
mouseEventArgs.LeftButton == MouseButtonState.Pressed ||
mouseEventArgs.RightButton == MouseButtonState.Pressed ||
mouseEventArgs.MiddleButton == MouseButtonState.Pressed ||
mouseEventArgs.XButton1 == MouseButtonState.Pressed ||
mouseEventArgs.XButton2 == MouseButtonState.Pressed
|| _inactiveMousePosition != mouseEventArgs.GetPosition(this)
))
{
return;
}
}
// set UI on activity
await System.Windows.Application.Current.Dispatcher.InvokeAsync(new Action(() =>
{
videoplayer.Stop();
Grid1.Visibility = Visibility.Visible;
Grid2.Visibility = Visibility.Hidden;
}));
_activityTimer.Stop();
_activityTimer.Start();
}
}
我不知道是否还有其他事件您应该过滤掉,我不知道通过事件名称过滤是最好的方法。