我有一个WPF窗口,由于部分原因,在窗口的初始化里面不能设置Topmost=true,需要在Loaded事件里面设置:
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
this.Loaded+= OnLoaded;
}
private void OnLoaded(object sender, RoutedEventArgs e)
{
this.Topmost = true;
}
}
在功能发布上线后,收到部分反馈说窗口没有置顶,这让我对微软的方法产生了怀疑,一度认为是win10,win11系统频繁更新而出现的bug。
在这个问题出现了多次后,我们认真进行了分析,然后我们猜测是否是在设置Topmost的时候,窗口处于没有焦点的状态,导致方法设置失败呢 ,基于如此,我们做了下面的实验:
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
this.Loaded+= OnLoaded;
}
private async void OnLoaded(object sender, RoutedEventArgs e)
{
await Task.Delay(3000);
this.Topmost = true;
}
}
很遗憾,采用这个方法仍然没有发现问题,后面想到可能是由于我们的项目是采用多进程窗口的方式,这个特殊性导致方法没有生效,最后我们通过以下的方式稳定复现了问题:
- 启动程序后,在窗口没有显示出来的时候将鼠标焦点切换到其它窗口(无限制,任何窗口都可,只要不是当前程序的窗口)
- 带程序窗口显示出来后,在测试窗口的置顶,这时候置顶就没有生效了
- 当手动点击窗口让窗口变为活动窗口后,在设置Topmost就能够生效
基于这个情况,让我想起了之前为了解决窗口打开时不显示的问题,感觉他们是一个问题:
WPF跨进程显示窗口,解决部分窗口不置顶或者不前置问题
那根据这个思路,我猜测出现这个问题的原因如下:
- 在窗口打开的过程中,由于鼠标点击了其它窗口,导致鼠标所在的线程和窗口当前的线程不一致
- 这时候设置Topmost=true时,方法没有报错,但是方法也没有生效
- 需要采用附加线程的方法,重新设置,然后再设置Topmost
基于猜测,然后对程序进行了如下修改:
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
this.Loaded+= OnLoaded;
}
private async void OnLoaded(object sender, RoutedEventArgs e)
{
this.ActivateWindow();
this.Topmost = true;
}
}
然后再进行测试,发现没有置顶的问题得到了解决。