如何防止在同一个应用程序中从窗口窃取焦点?

时间:2015-08-10 14:05:34

标签: c# winapi

使用SetWindowLong()WS_EX_NOACTIVATE我已经创建了一个不可激活的窗口。

  • 如果我将光标放在记事本中并单击不可激活的窗口,则光标在记事本中保持活动状态。
  • 如果我将光标放在应用程序内的另一个窗口中并单击不可激活的窗口,光标将从另一个窗口中删除。

我做了一个小型项目来重现这个问题:http://s000.tinyupload.com/index.php?file_id=78888532457762447347

基本上是一个带有两个窗口的WPF应用程序:

主窗口

// XAML code
<Window x:Class="CS_test.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Background="LightGray" Height="300" Width="400">
    <Grid>
        <TextBox Width="300" Height="200"></TextBox>
    </Grid>
</Window>

// C# code
namespace CS_test
{
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
            new NonActiveWindow().Show();
        }
    }
}

NonActiveWindow

// XAML code
<Window x:Class="CS_test.NonActiveWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="NonActiveWindow" Height="300" Width="300"
        Loaded="Window_Loaded">
    <Grid>
    </Grid>
</Window>

// C# code
namespace CS_test
{
    public partial class NonActiveWindow : Window
    {
        [DllImport("user32.dll")]
        public static extern IntPtr SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);

        [DllImport("user32.dll")]
        public static extern int GetWindowLong(IntPtr hWnd, int nIndex);

        private const int GWL_EXSTYLE = -20;
        private const int WS_EX_NOACTIVATE = 0x08000000;

        public NonActiveWindow()
        {
            InitializeComponent();
        }

        private void Window_Loaded(object sender, RoutedEventArgs e)
        {
            WindowInteropHelper windowHelper = new WindowInteropHelper(this);
            SetWindowLong(windowHelper.Handle, GWL_EXSTYLE, GetWindowLong(windowHelper.Handle, GWL_EXSTYLE) | WS_EX_NOACTIVATE);
        }
    }
}

我知道可以通过在单独的线程(使用单独的Dispatcher)中实例化NonActiveWindow来解决问题,但由于线程问题,我无法使用此方法。

有没有办法防止焦点丢失而不在单独的线程中创建不可激活的窗口?

编辑:

视频显示问题(回应Justins的回答):

https://www.youtube.com/watch?v=6RNmJc7ya48

以防万一有人想知道拖动时窗口没有更新的原因:Create a window using the WS_EX_NOACTIVATE flag ,but it can't be dragged until I release the mouse

1 个答案:

答案 0 :(得分:0)

ShowActivated属性对我来说很好:

public MainWindow()
{
    InitializeComponent();
    var otherWindow = new NonActiveWindow()
    {
        ShowActivated = false
    };
    otherWindow.Show();
}

我的OtherWindow只是一个空的WPF窗口,并且没有您的示例所拥有的任何P / Invoke调用。

请注意,在显示 MainWindow之后,OtherWindow会稍微显示,因为此调用位于构造函数中。如果你想解决这个问题,那么你应该在构造函数之外调用Show,例如在Loaded事件中。