本文记录我阅读 WPF 项目了解到的 ShowInTaskbar 的实现原理
在 WPF 里面,通过判断 ShowInTaskbar == false 决定是否在任务栏隐藏窗口图标,其实现原理是创建一个隐藏的窗口,将当前窗口的 Owner 设置为此隐藏窗口而实现
在 Window.EnsureHiddenWindow 方法里面将创建名为 Hidden Window 的隐藏窗口,这个隐藏窗口的唯一作用就是在 ShowInTaskbar == false 时,被设置为当前窗口的 Owner 从而实现隐藏窗口的功能
/// <summary>
/// Get or create the hidden window used for parenting when ShowInTaskbar == false.
/// </summary>
private HwndWrapper EnsureHiddenWindow()
{
if (_hiddenWindow == null)
{
_hiddenWindow = new HwndWrapper(
0, // classStyle
NativeMethods.WS_OVERLAPPEDWINDOW, // style
0, // exStyle
NativeMethods.CW_USEDEFAULT, // x
NativeMethods.CW_USEDEFAULT, // y
NativeMethods.CW_USEDEFAULT, // width
NativeMethods.CW_USEDEFAULT, // height
"Hidden Window", // name
IntPtr.Zero,
null
);
}
return _hiddenWindow;
}
在 ShowInTaskbar 变更的时候,将先调用 EnsureHiddenWindow 方法让 _hiddenWindow 完成执行初始化,而后再调用 SetOwnerHandle(_hiddenWindow.Handle); 将隐藏窗口为当前的窗口的 Owner 从而实现在任务栏隐藏窗口图标的功能
/// <summary>
/// The DependencyProperty for ShowInTaskbarProperty.
/// Flags: None
/// Default Value: true
/// </summary>
public static readonly DependencyProperty ShowInTaskbarProperty =
DependencyProperty.Register("ShowInTaskbar",
typeof(bool),
typeof(Window),
new FrameworkPropertyMetadata(BooleanBoxes.TrueBox,
new PropertyChangedCallback(_OnShowInTaskbarChanged),
new CoerceValueCallback(VerifyAccessCoercion)));
private static void _OnShowInTaskbarChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
Window w = (Window)d;
Debug.Assert(w != null, "DependencyObject must be of type Window.");
w.OnShowInTaskbarChanged();
}
private void OnShowInTaskbarChanged()
{
...
SetTaskbarStatus();
...
}
/// <summary>
/// sets taskbar status
/// </summary>
private void SetTaskbarStatus()
{
if (ShowInTaskbar == false) // don't show in taskbar
{
// To remove the taskbar button for this window it needs to have a non-null parent
// (we'll create a hidden window for this purpose) and not have WS_EX_APPWINDOW
// Create this now, even if we're not currently going to parent it.
// If the Owner changes, we'll need to switch to this.
EnsureHiddenWindow();
// when this window is unowned
if (_ownerHandle == IntPtr.Zero)
{
SetOwnerHandle(_hiddenWindow.Handle);
...
}
...
}
else // (ShowInTaskbar == true) show in task bar
{
...
}
}
更多关于 ShowInTaskbar 请参阅:
WPF 设置 ShowInTaskbar 对窗口最小化的影响
当前的 WPF 在 https://github.com/dotnet/wpf 完全开源,使用友好的 MIT 协议,意味着允许任何人任何组织和企业任意处置,包括使用,复制,修改,合并,发表,分发,再授权,或者销售。在仓库里面包含了完全的构建逻辑,只需要本地的网络足够好(因为需要下载一堆构建工具),即可进行本地构建
本文会经常更新,请阅读原文: https://blog.lindexi.com/post/dotnet-%E8%AF%BB-WPF-%E6%BA%90%E4%BB%A3%E7%A0%81%E7%AC%94%E8%AE%B0-ShowInTaskbar-%E7%9A%84%E5%AE%9E%E7%8E%B0%E5%8E%9F%E7%90%86.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
如果你想持续阅读我的最新博客,请点击 RSS 订阅,推荐使用RSS Stalker订阅博客,或者收藏我的博客导航
本作品采用
知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议
进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接:
https://blog.lindexi.com
),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请
与我联系
。
无盈利,不卖课,做纯粹的技术博客
以下是广告时间
推荐关注 Edi.Wang 的公众号
欢迎进入 Eleven 老师组建的 .NET 社区
以上广告全是友情推广,无盈利