WPF中全屏窗口,會自動隱藏任務欄。
那非全屏窗口如何隱藏任務欄?甚至有沒有一種場景,隱藏任務欄后自定義一套系統任務欄來顯示?
以下會分階段講述一些概念,任務欄、查找窗口、控制窗口顯示。
1. 主屏任務欄
任務欄,其實也是一個窗口,主屏的任務欄名稱是"Shell_TrayWnd"。
所以可以通過名稱查找窗口,然后對窗口進行顯示、隱藏操作。
以下是主屏幕任務欄的控制操作:
1 public static class ScreenTaskBar 2 { 3 private const int SwHide = 0; //隱藏窗口 4 private const int SwRestore = 9;//還原窗口 5 6 [DllImport("user32.dll")] 7 private static extern int ShowWindow(int hwnd, int nCmdShow); 8 [DllImport("user32.dll")] 9 private static extern int FindWindow(string lpClassName, string lpWindowName); 10 /// <summary> 11 /// 顯示任務欄 12 /// </summary> 13 public static void Show() 14 { 15 ShowWindow(FindWindow("Shell_TrayWnd", null), SwRestore); 16 } 17 /// <summary> 18 /// 隱藏任務欄 19 /// </summary> 20 public static void Hide() 21 { 22 ShowWindow(FindWindow("Shell_TrayWnd", null), SwHide); 23 } 24 }
2. 多屏任務欄
如果是多屏,對任務欄進行處理的場景,一般是對窗口所對應的任務欄操作。
如何獲取任意窗口所在的任務欄呢?既然任務欄也是窗口,那么我們的關注點就是如何找到任務欄窗口了。
User32有EnumWindows函數,可以遍歷當前電腦的所有窗口。
1 private delegate bool EnumWindowProc(IntPtr hWnd, int lParam); 2 [DllImport("user32")] 3 private static extern bool EnumWindows(EnumWindowProc lpEnumFunc, int lParam);
EnumWindowProc定義了委托的處理。添加回調方法,返回的參數是句柄信息:
1 bool OnEnumWindow(IntPtr hWnd, int lparam) 2 { 3 //添加代碼XXX 4 return true; 5 }
然后可以在回調內部添加代碼,根據窗口的句柄信息,我們去拿窗口的一些信息,類名、窗口標題、窗口的Bounds(位置、大小)
1 [DllImport("user32")] 2 private static extern int GetClassName(IntPtr hWnd, StringBuilder lpString, int nMaxCount); 3 [DllImport("user32")] 4 private static extern int GetWindowText(IntPtr hWnd, StringBuilder lptrString, int nMaxCount); 5 [DllImport("user32")] 6 private static extern bool GetWindowRect(IntPtr hWnd, ref LPRECT rect);
以下是部分遍歷出來的窗口類信息:

所以,可以篩選出那些以TrayWnd字符串結尾的,這些都是任務欄窗口。
之后就是如何篩選出我們想要的任務欄,即窗口對應的任務欄。
窗口與任務欄,是通過屏幕關聯在一起的。通過窗口獲取當前屏幕信息,任務欄的Bounds如果與屏幕Bounds相交,則說明此任務欄在此屏幕內。
1 var intPtr = new WindowInteropHelper(window).Handle;//獲取當前窗口的句柄 2 var screen = Screen.FromHandle(intPtr);//獲取當前屏幕 3 var currentScreenBounds = screen.Bounds; 4 var taskBars = windows.Where(i => i.ClassName.Contains("TrayWnd")); 5 var currentTaskBar = taskBars.FirstOrDefault(i => i.Bounds.IntersectsWith(currentScreenBounds));
獲取任務欄,也可以通過任務欄的句柄獲取屏幕,與主窗口所在屏幕判斷是否同一個。
獲取了指定的任務欄信息后,我們就可以控制任務欄顯示、隱藏了。調用下user32下函數ShowWindow即可:
1 private const int SwHide = 0; //隱藏窗口 2 private const int SwRestore = 9;//還原窗口 3 /// <summary> 4 /// 通過句柄,窗體顯示函數 5 /// </summary> 6 /// <param name="hWnd">窗體句柄</param> 7 /// <param name="cmdShow">顯示方式</param> 8 /// <returns>返回成功與否</returns> 9 [DllImport("user32.dll", EntryPoint = "ShowWindowAsync", SetLastError = true)] 10 public static extern bool ShowWindow(IntPtr hWnd, int cmdShow);
這里的ShowWindow,與上面默認任務欄操作所調用的ShowWindow不一樣,句柄參數是IntPtr
