using System; using System.Collections.Generic; using System.Runtime.InteropServices; using System.Text; // 由衷感謝 lindexi 提供的 《WPF 稳定的全屏化窗口方法》 // 文章鏈接:https://blog.lindexi.com/post/WPF-%E7%A8%B3%E5%AE%9A%E7%9A%84%E5%85%A8%E5%B1%8F%E5%8C%96%E7%AA%97%E5%8F%A3%E6%96%B9%E6%B3%95.html // lindexi 的部落格:https://blog.lindexi.com/ namespace Ink_Canvas.Helpers { public static partial class FullScreenHelper { static class Win32 { [Flags] public enum ShowWindowCommands { /// /// Maximizes the specified window. /// SW_MAXIMIZE = 3, /// /// Activates and displays the window. If the window is minimized or maximized, the system restores it to its original /// size and position. An application should specify this flag when restoring a minimized window. /// SW_RESTORE = 9, } internal static class Properties { #if !ANSI public const CharSet BuildCharSet = CharSet.Unicode; #else public const CharSet BuildCharSet = CharSet.Ansi; #endif } public static class Dwmapi { public const string LibraryName = "Dwmapi.dll"; [DllImport(LibraryName, ExactSpelling = true, PreserveSig = false)] [return: MarshalAs(UnmanagedType.Bool)] public static extern bool DwmIsCompositionEnabled(); [DllImport("Dwmapi.dll", ExactSpelling = true, SetLastError = true)] public static extern int DwmSetWindowAttribute(IntPtr hwnd, DWMWINDOWATTRIBUTE dwAttribute, in int pvAttribute, uint cbAttribute); } public static class User32 { public const string LibraryName = "user32"; [DllImport(LibraryName, CharSet = Properties.BuildCharSet)] public static extern bool GetMonitorInfo(IntPtr hMonitor, ref MonitorInfo lpmi); [DllImport(LibraryName, ExactSpelling = true)] public static extern IntPtr MonitorFromRect(in Rectangle lprc, MonitorFlag dwFlags); [DllImport(LibraryName, ExactSpelling = true)] public static extern bool IsIconic(IntPtr hwnd); [DllImport(LibraryName, ExactSpelling = true)] public static extern bool ShowWindow(IntPtr hWnd, ShowWindowCommands nCmdShow); [DllImport(LibraryName, ExactSpelling = true)] public static extern bool SetWindowPlacement(IntPtr hWnd, [In] ref WINDOWPLACEMENT lpwndpl); [return: MarshalAs(UnmanagedType.Bool)] [DllImport(LibraryName, ExactSpelling = true)] public static extern bool GetWindowRect(IntPtr hWnd, out Rectangle lpRect); [DllImport(LibraryName, ExactSpelling = true, SetLastError = true)] public static extern Int32 SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, Int32 x, Int32 y, Int32 cx, Int32 cy, Int32 wFlagslong); [DllImport(LibraryName, ExactSpelling = true)] public static extern bool GetWindowPlacement(IntPtr hWnd, ref WINDOWPLACEMENT lpwndpl); public static IntPtr GetWindowLongPtr(IntPtr hWnd, GetWindowLongFields nIndex) => GetWindowLongPtr(hWnd, (int) nIndex); public static IntPtr GetWindowLongPtr(IntPtr hWnd, int nIndex) { return IntPtr.Size > 4 #pragma warning disable CS0618 // 类型或成员已过时 ? GetWindowLongPtr_x64(hWnd, nIndex) : new IntPtr(GetWindowLong(hWnd, nIndex)); #pragma warning restore CS0618 // 类型或成员已过时 } [DllImport(LibraryName, CharSet = Properties.BuildCharSet)] public static extern int GetWindowLong(IntPtr hWnd, int nIndex); [DllImport(LibraryName, CharSet = Properties.BuildCharSet, EntryPoint = "GetWindowLongPtr")] public static extern IntPtr GetWindowLongPtr_x64(IntPtr hWnd, int nIndex); public static IntPtr SetWindowLongPtr(IntPtr hWnd, GetWindowLongFields nIndex, IntPtr dwNewLong) => SetWindowLongPtr(hWnd, (int) nIndex, dwNewLong); public static IntPtr SetWindowLongPtr(IntPtr hWnd, int nIndex, IntPtr dwNewLong) { return IntPtr.Size > 4 #pragma warning disable CS0618 // 类型或成员已过时 ? SetWindowLongPtr_x64(hWnd, nIndex, dwNewLong) : new IntPtr(SetWindowLong(hWnd, nIndex, dwNewLong.ToInt32())); #pragma warning restore CS0618 // 类型或成员已过时 } [DllImport(LibraryName, CharSet = Properties.BuildCharSet, EntryPoint = "SetWindowLongPtr")] public static extern IntPtr SetWindowLongPtr_x64(IntPtr hWnd, int nIndex, IntPtr dwNewLong); [DllImport(LibraryName, CharSet = Properties.BuildCharSet)] public static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong); } } [StructLayout(LayoutKind.Sequential)] struct MonitorInfo { /// /// The size of the structure, in bytes. /// public uint Size; /// /// A RECT structure that specifies the display monitor rectangle, expressed in virtual-screen coordinates. Note that /// if the monitor is not the primary display monitor, some of the rectangle's coordinates may be negative values. /// public Rectangle MonitorRect; /// /// A RECT structure that specifies the work area rectangle of the display monitor, expressed in virtual-screen /// coordinates. Note that if the monitor is not the primary display monitor, some of the rectangle's coordinates may /// be negative values. /// public Rectangle WorkRect; /// /// A set of flags that represent attributes of the display monitor. /// public MonitorInfoFlag Flags; } enum MonitorInfoFlag { } enum MonitorFlag { /// /// Returns a handle to the primary display monitor. /// MONITOR_DEFAULTTOPRIMARY = 1, } [StructLayout(LayoutKind.Sequential)] struct WindowPosition { public IntPtr Hwnd; public IntPtr HwndZOrderInsertAfter; public int X; public int Y; public int Width; public int Height; public WindowPositionFlags Flags; } enum HwndZOrder { /// /// Places the window at the top of the Z order. /// HWND_TOP = 0, } enum DWMWINDOWATTRIBUTE : uint { DWMWA_TRANSITIONS_FORCEDISABLED = 3, } enum GetWindowLongFields { /// /// 设定一个新的窗口风格 /// Retrieves the window styles /// GWL_STYLE = -16, } [StructLayout(LayoutKind.Sequential)] struct WINDOWPLACEMENT // WindowPlacement { public uint Size; public WindowPlacementFlags Flags; public Win32.ShowWindowCommands ShowCmd; public Point MinPosition; public Point MaxPosition; public Rectangle NormalPosition; } [Flags] public enum WindowPositionFlags { /// /// /// 清除客户区的所有内容。如果未设置该标志,客户区的有效内容被保存并且在窗口尺寸更新和重定位后拷贝回客户区 /// /// Discards the entire contents of the client area. If this flag is not specified, the valid contents of the client /// area are saved and copied back into the client area after the window is sized or repositioned. /// SWP_NOCOPYBITS = 0x0100, /// /// /// 维持当前位置(忽略X和Y参数) /// /// Retains the current position (ignores X and Y parameters). /// SWP_NOMOVE = 0x0002, /// /// /// 不重画改变的内容。如果设置了这个标志,则不发生任何重画动作。适用于客户区和非客户区(包括标题栏和滚动条)和任何由于窗回移动而露出的父窗口的所有部分。如果设置了这个标志,应用程序必须明确地使窗口无效并区重画窗口的任何部分和父窗口需要重画的部分 /// /// Does not redraw changes. If this flag is set, no repainting of any kind occurs. This applies to the client area, /// the nonclient area (including the title bar and scroll bars), and any part of the parent window uncovered as a /// result of the window being moved. When this flag is set, the application must explicitly invalidate or redraw any /// parts of the window and parent window that need redrawing. /// SWP_NOREDRAW = 0x0008, /// /// /// 维持当前尺寸(忽略 cx 和 cy 参数) /// /// Retains the current size (ignores the cx and cy parameters). /// SWP_NOSIZE = 0x0001, /// /// /// 维持当前 Z 序(忽略 hWndlnsertAfter 参数) /// /// Retains the current Z order (ignores the hWndInsertAfter parameter). /// SWP_NOZORDER = 0x0004, } [StructLayout(LayoutKind.Sequential)] struct Rectangle { public int Left; public int Top; public int Right; public int Bottom; /// /// 矩形的宽度 /// public int Width { get { return unchecked((int) (Right - Left)); } set { Right = unchecked((int) (Left + value)); } } /// /// 矩形的高度 /// public int Height { get { return unchecked((int) (Bottom - Top)); } set { Bottom = unchecked((int) (Top + value)); } } public bool Equals(Rectangle other) { return (Left == other.Left) && (Right == other.Right) && (Top == other.Top) && (Bottom == other.Bottom); } public override bool Equals(object obj) { return obj is Rectangle rectangle && Equals(rectangle); } public static bool operator ==(Rectangle left, Rectangle right) { return left.Equals(right); } public override int GetHashCode() { unchecked { var hashCode = (int) Left; hashCode = (hashCode * 397) ^ (int) Top; hashCode = (hashCode * 397) ^ (int) Right; hashCode = (hashCode * 397) ^ (int) Bottom; return hashCode; } } public static bool operator !=(Rectangle left, Rectangle right) { return !(left == right); } } [StructLayout(LayoutKind.Sequential)] struct Point { public int X; public int Y; } [Flags] enum WindowPlacementFlags { } [Flags] enum WindowStyles { /// /// The window is initially maximized. /// WS_MAXIMIZE = 0x01000000, /// /// The window has a maximize button. Cannot be combined with the WS_EX_CONTEXTHELP style. The WS_SYSMENU style must /// also be specified. /// WS_MAXIMIZEBOX = 0x00010000, /// /// The window is initially minimized. Same as the WS_ICONIC style. /// WS_MINIMIZE = 0x20000000, /// /// The window has a sizing border. Same as the WS_SIZEBOX style. /// WS_THICKFRAME = 0x00040000, } [ComImport] [Guid("602D4995-B13A-429b-A66E-1935E44F4317")] [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] private interface ITaskbarList2 { [PreserveSig] int HrInit(); [PreserveSig] int AddTab(IntPtr hwnd); [PreserveSig] int DeleteTab(IntPtr hwnd); [PreserveSig] int ActivateTab(IntPtr hwnd); [PreserveSig] int SetActiveAlt(IntPtr hwnd); [PreserveSig] int MarkFullscreenWindow(IntPtr hwnd, [MarshalAs(UnmanagedType.Bool)] bool fFullscreen); } } }