使用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。
答案 0 :(得分:0)
ShowActivated
属性对我来说很好:
public MainWindow()
{
InitializeComponent();
var otherWindow = new NonActiveWindow()
{
ShowActivated = false
};
otherWindow.Show();
}
我的OtherWindow
只是一个空的WPF窗口,并且没有您的示例所拥有的任何P / Invoke调用。
请注意,在显示 MainWindow
之后,OtherWindow
会稍微显示,因为此调用位于构造函数中。如果你想解决这个问题,那么你应该在构造函数之外调用Show
,例如在Loaded
事件中。