Files

745 lines
35 KiB
C#
Raw Permalink Normal View History

2025-08-23 21:39:21 +08:00
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Drawing;
using System.Drawing.Imaging;
using System.IO;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Documents;
using System.Windows.Interop;
using System.Windows.Media.Imaging;
using OSVersionExtension;
using Vanara.PInvoke;
using Encoder = System.Drawing.Imaging.Encoder;
using OperatingSystem = OSVersionExtension.OperatingSystem;
using PixelFormat = System.Drawing.Imaging.PixelFormat;
using System.Management;
using System.Reflection;
using System.Windows.Shapes;
using Path = System.IO.Path;
using Rectangle = System.Drawing.Rectangle;
using Ink_Canvas.Helpers;
namespace Ink_Canvas {
public partial class MainWindow : PerformanceTransparentWin {
#region MagnificationAPI ICC窗口
#region Dubi906w
[DllImport("user32.dll", EntryPoint = "SetWindowLong")]
private static extern int SetWindowLong32(IntPtr hWnd, int nIndex, int dwNewLong);
[DllImport("user32.dll", EntryPoint = "SetWindowLongPtr")]
private static extern IntPtr SetWindowLongPtr64(IntPtr hWnd, int nIndex, IntPtr dwNewLong);
[DllImport("user32.dll", EntryPoint = "GetWindowLong")]
private static extern IntPtr GetWindowLongPtr(IntPtr hWnd, int nIndex);
[DllImport("user32.dll")]
static extern bool ShowWindow(IntPtr hWnd, int nCmdShow);
#endregion Dubi906w
#region Win32 AlanCRL
// 感謝 Alan-CRL 造的輪子
private const int WS_EX_TOPMOST = 0x00000008;
private const int WS_EX_LAYERED = 0x00080000;
private const int WS_SIZEBOX = 0x00040000;
private const int WS_SYSMENU = 0x00080000;
private const int WS_CLIPCHILDREN = 0x02000000;
private const int WS_CAPTION = 0x00C00000;
private const int WS_MAXIMIZEBOX = 0x00010000;
private const int GWL_STYLE = -16;
private const int GWL_EXSTYLE = -20;
private const int WS_THICKFRAME = 0x00040000;
private const int SWP_NOSIZE = 0x0001;
private const int SWP_FRAMECHANGED = 0x0020;
private const int WS_EX_TOOLWINDOW = 0x00000080;
private const int WS_EX_APPWINDOW = 0x00040000;
private const int SW_SHOW = 5;
private const int LWA_ALPHA = 0x00000002;
private const int PW_RENDERFULLCONTENT = 2;
private static IntPtr windowHostHandle;
// PInvoke 輪子
[DllImport("user32.dll", SetLastError = true)]
private static extern IntPtr CreateWindowEx(int dwExStyle, string lpClassName, string lpWindowName, int dwStyle,
int x, int y, int nWidth, int nHeight, IntPtr hWndParent, IntPtr hMenu, IntPtr hInstance, IntPtr lpParam);
[DllImport("user32.dll")]
private static extern bool DestroyWindow(IntPtr hWnd);
[DllImport("user32.dll", SetLastError = true)]
private static extern ushort RegisterClassEx(ref WNDCLASSEX lpwcx);
[DllImport("user32.dll", SetLastError = true)]
private static extern IntPtr DefWindowProc(IntPtr hWnd, uint msg, IntPtr wParam, IntPtr lParam);
[DllImport("user32.dll", SetLastError = true)]
private static extern IntPtr LoadCursor(IntPtr hInstance, int lpCursorName);
[DllImport("user32.dll", SetLastError = true)]
private static extern short UnregisterClass(string lpClassName, IntPtr hInstance);
[DllImport("user32.dll", SetLastError = true)]
private static extern bool SetLayeredWindowAttributes(IntPtr hwnd, uint crKey, byte bAlpha, uint dwFlags);
[DllImport("user32.dll", SetLastError = true)]
private static extern int GetWindowLong(IntPtr hWnd, int nIndex);
[DllImport("user32.dll", SetLastError = true)]
private static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);
[DllImport("user32.dll", SetLastError = true)]
private static extern bool InvalidateRect(IntPtr hWnd, IntPtr lpRect, bool bErase);
[StructLayout(LayoutKind.Sequential)]
private struct WNDCLASSEX {
public uint cbSize;
public uint style;
[MarshalAs(UnmanagedType.FunctionPtr)] public WndProc lpfnWndProc;
public int cbClsExtra;
public int cbWndExtra;
public IntPtr hInstance;
public IntPtr hIcon;
public IntPtr hCursor;
public IntPtr hbrBackground;
public string lpszMenuName;
public string lpszClassName;
public IntPtr hIconSm;
}
private delegate IntPtr WndProc(IntPtr hWnd, uint msg, IntPtr wParam, IntPtr lParam);
private static readonly WndProc StaticWndProcDelegate = WndHostProc;
private const uint WM_DESTROY = 0x0002;
private const uint WM_CLOSE = 0x0010;
private const int CS_HREDRAW = 0x0002;
private const int CS_VREDRAW = 0x0001;
private const int IDC_ARROW = 32512;
private static int COLOR_BTNFACE = 15;
private const int WS_CHILD = 0x40000000;
private const int WS_VISIBLE = 0x10000000;
private const int MS_CLIPAROUNDCURSOR = 0x0002;
#endregion Win32 AlanCRL
public void SaveScreenshotToDesktopByMagnificationAPI(HWND[] hwndsList,
Action<Bitmap> callbackAction, bool isUsingCallback = false) {
if (OSVersion.GetOperatingSystem() < OperatingSystem.Windows81) return;
if (!Magnification.MagInitialize()) return;
// 註冊宿主窗體類名
var wndClassEx = new WNDCLASSEX {
cbSize = (uint)Marshal.SizeOf<WNDCLASSEX>(), style = CS_HREDRAW | CS_VREDRAW,
lpfnWndProc = StaticWndProcDelegate, hInstance = IntPtr.Zero,
hCursor = LoadCursor(IntPtr.Zero, IDC_ARROW), hbrBackground = (IntPtr)(1 + COLOR_BTNFACE),
lpszClassName = "ICCMagnifierWindowHost",
hIcon = IntPtr.Zero, hIconSm = IntPtr.Zero
};
RegisterClassEx(ref wndClassEx);
// 創建宿主窗體
windowHostHandle = CreateWindowEx(
WS_EX_TOPMOST | WS_EX_LAYERED, "ICCMagnifierWindowHost", "ICCMagnifierWindowHostWindow",
WS_SIZEBOX | WS_SYSMENU | WS_CLIPCHILDREN | WS_CAPTION | WS_MAXIMIZEBOX, 0, 0,
System.Windows.Forms.Screen.PrimaryScreen.Bounds.Width,
System.Windows.Forms.Screen.PrimaryScreen.Bounds.Height, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero,
IntPtr.Zero);
// 設定分層窗體
SetLayeredWindowAttributes(windowHostHandle, 0, 0, LWA_ALPHA);
// 創建放大鏡窗體
var hwndMag = CreateWindowEx(
0, Magnification.WC_MAGNIFIER, "ICCMagnifierWindow", WS_CHILD | WS_VISIBLE | MS_CLIPAROUNDCURSOR, 0, 0,
System.Windows.Forms.Screen.PrimaryScreen.Bounds.Width,
System.Windows.Forms.Screen.PrimaryScreen.Bounds.Height, windowHostHandle,
IntPtr.Zero, IntPtr.Zero, IntPtr.Zero);
// 設定窗體樣式和排布
int style = GetWindowLong(windowHostHandle, GWL_STYLE);
style &= ~WS_CAPTION; // 隐藏标题栏
style &= ~WS_THICKFRAME; // 禁止窗口拉伸
SetWindowLong(windowHostHandle, GWL_STYLE, style);
SetWindowPos(windowHostHandle, IntPtr.Zero, 0, 0, 0, 0, SWP_NOSIZE | SWP_FRAMECHANGED);
// 設定額外樣式
int exStyle = GetWindowLong(windowHostHandle, GWL_EXSTYLE);
exStyle |= WS_EX_TOOLWINDOW; /* <- 隐藏任务栏图标 */
exStyle &= ~WS_EX_APPWINDOW;
SetWindowLong(windowHostHandle, GWL_EXSTYLE, exStyle);
// 設定放大鏡工廠
Magnification.MAGTRANSFORM matrix = new Magnification.MAGTRANSFORM();
matrix[0, 0] = 1.0f;
matrix[0, 1] = 0.0f;
matrix[0, 2] = 0.0f;
matrix[1, 0] = 0.0f;
matrix[1, 1] = 1.0f;
matrix[1, 2] = 0.0f;
matrix[2, 0] = 1.0f;
matrix[2, 1] = 0.0f;
matrix[2, 2] = 0.0f;
if (!Magnification.MagSetWindowTransform(hwndMag, matrix)) return;
// 設定放大鏡轉化矩乘陣列
Magnification.MAGCOLOREFFECT magEffect = new Magnification.MAGCOLOREFFECT();
magEffect[0, 0] = 1.0f;
magEffect[0, 1] = 0.0f;
magEffect[0, 2] = 0.0f;
magEffect[0, 3] = 0.0f;
magEffect[0, 4] = 0.0f;
magEffect[1, 0] = 0.0f;
magEffect[1, 1] = 1.0f;
magEffect[1, 2] = 0.0f;
magEffect[1, 3] = 0.0f;
magEffect[1, 4] = 0.0f;
magEffect[2, 0] = 0.0f;
magEffect[2, 1] = 0.0f;
magEffect[2, 2] = 1.0f;
magEffect[2, 3] = 0.0f;
magEffect[2, 4] = 0.0f;
magEffect[3, 0] = 0.0f;
magEffect[3, 1] = 0.0f;
magEffect[3, 2] = 0.0f;
magEffect[3, 3] = 1.0f;
magEffect[3, 4] = 0.0f;
magEffect[4, 0] = 0.0f;
magEffect[4, 1] = 0.0f;
magEffect[4, 2] = 0.0f;
magEffect[4, 3] = 0.0f;
magEffect[4, 4] = 1.0f;
if (!Magnification.MagSetColorEffect(hwndMag, magEffect)) return;
// 顯示窗體
ShowWindow(windowHostHandle, SW_SHOW);
// 过滤窗口
var hwnds = new List<HWND> { hwndMag };
hwnds.AddRange(hwndsList);
if (!Magnification.MagSetWindowFilterList(hwndMag, Magnification.MW_FILTERMODE.MW_FILTERMODE_EXCLUDE,
hwnds.Count, hwnds.ToArray())) return;
// 设置窗口 Source
if (!Magnification.MagSetWindowSource(hwndMag, new RECT(0, 0,
System.Windows.Forms.Screen.PrimaryScreen.Bounds.Width,
System.Windows.Forms.Screen.PrimaryScreen.Bounds.Height))) return;
InvalidateRect(hwndMag, IntPtr.Zero, true);
// 抓取屏幕圖像
if (isUsingCallback) {
if (!Magnification.MagSetImageScalingCallback(hwndMag,
(hwnd, srcdata, srcheader, destdata, destheader, unclipped, clipped, dirty) => {
Bitmap bm = new Bitmap((int)srcheader.width, (int)srcheader.height,
(int)srcheader.width * 4, PixelFormat.Format32bppRgb, srcdata);
callbackAction(bm);
return true;
})) return;
} else {
RECT rect;
GetWindowRect(hwndMag, out rect);
Bitmap bmp = new Bitmap(rect.Width, rect.Height);
Graphics memoryGraphics = Graphics.FromImage(bmp);
PrintWindow(hwndMag, memoryGraphics.GetHdc(), PW_RENDERFULLCONTENT);
memoryGraphics.ReleaseHdc();
callbackAction(bmp);
}
// 反注册宿主窗口
UnregisterClass("ICCMagnifierWindowHost", IntPtr.Zero);
// 销毁宿主窗口
Magnification.MagUninitialize();
DestroyWindow(windowHostHandle);
}
public Task<Bitmap> SaveScreenshotToDesktopByMagnificationAPIAsync(HWND[] hwndsList,
bool isUsingCallback = false) {
return Task.Run(() => {
var t = new TaskCompletionSource<Bitmap>();
SaveScreenshotToDesktopByMagnificationAPI(hwndsList, bitmap => t.TrySetResult(bitmap), isUsingCallback);
return t.Task;
});
}
private static IntPtr WndHostProc(IntPtr hWnd, uint msg, IntPtr wParam, IntPtr lParam) {
return DefWindowProc(hWnd, msg, wParam, lParam);
}
#endregion MagnificationAPI ICC窗口
#region Powerpoint
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = false)]
static extern IntPtr SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
public static IntPtr GetClassLongPtr(IntPtr hWnd, int nIndex) {
if (IntPtr.Size > 4)
return GetClassLongPtr64(hWnd, nIndex);
else
return new IntPtr(GetClassLongPtr32(hWnd, nIndex));
}
[DllImport("user32.dll", EntryPoint = "GetClassLong")]
public static extern uint GetClassLongPtr32(IntPtr hWnd, int nIndex);
[DllImport("user32.dll", EntryPoint = "GetClassLongPtr")]
public static extern IntPtr GetClassLongPtr64(IntPtr hWnd, int nIndex);
[DllImport("user32.dll", EntryPoint = "EnumDesktopWindows",
ExactSpelling = false, CharSet = CharSet.Auto, SetLastError = true)]
public static extern bool EnumDesktopWindows(IntPtr hDesktop, Delegate lpEnumCallbackFunction, IntPtr lParam);
[DllImport("user32.dll")]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool IsWindowVisible(IntPtr hWnd);
[DllImport("user32.dll", EntryPoint = "GetWindowText",
ExactSpelling = false, CharSet = CharSet.Auto, SetLastError = true)]
public static extern int GetWindowText(IntPtr hWnd, StringBuilder lpWindowText, int nMaxCount);
[DllImport("user32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool PrintWindow(IntPtr hwnd, IntPtr hDC, uint nFlags);
[DllImport("user32.dll")]
private static extern bool GetWindowRect(IntPtr handle, out RECT rect);
[DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
static extern int GetWindowTextLength(IntPtr hWnd);
[DllImport("user32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool GetWindowPlacement(IntPtr hWnd, ref WINDOWPLACEMENT lpwndpl);
[DllImport("user32.dll", CharSet = CharSet.Unicode)]
public static extern IntPtr FindWindowEx(IntPtr parentHandle, IntPtr childAfter, string lclassName,
string windowTitle);
[DllImport("user32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
public static extern IntPtr GetShellWindow();
[DllImport("dwmapi.dll")]
static extern int DwmGetWindowAttribute(IntPtr hwnd, int dwAttribute, out bool pvAttribute, int cbAttribute);
[DllImport("dwmapi.dll")]
static extern int DwmGetWindowAttribute(IntPtr hwnd, DwmWindowAttribute dwAttribute, out RECT pvAttribute, int cbAttribute);
[DllImport("user32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool GetLayeredWindowAttributes(IntPtr hwnd, out uint crKey, out byte bAlpha, out uint dwFlags);
public delegate bool EnumWindowsProc(IntPtr hwnd, IntPtr lParam);
[DllImport("user32.dll", SetLastError=true)]
static extern uint GetWindowThreadProcessId(IntPtr hWnd, out uint processId);
[DllImport("user32.dll")]
internal static extern int GetDpiForWindow(IntPtr hWnd);
enum DwmWindowAttribute : uint {
NCRenderingEnabled = 1,
NCRenderingPolicy,
TransitionsForceDisabled,
AllowNCPaint,
CaptionButtonBounds,
NonClientRtlLayout,
ForceIconicRepresentation,
Flip3DPolicy,
ExtendedFrameBounds,
HasIconicBitmap,
DisallowPeek,
ExcludedFromPeek,
Cloak,
Cloaked,
FreezeRepresentation,
PassiveUpdateMode,
UseHostBackdropBrush,
UseImmersiveDarkMode = 20,
WindowCornerPreference = 33,
BorderColor,
CaptionColor,
TextColor,
VisibleFrameBorderThickness,
SystemBackdropType,
Last
}
public Icon GetAppIcon(IntPtr hwnd) {
IntPtr iconHandle = SendMessage(hwnd, 0x7F, 2, 0);
if (iconHandle == IntPtr.Zero)
iconHandle = SendMessage(hwnd, 0x7F, 0, 0);
if (iconHandle == IntPtr.Zero)
iconHandle = SendMessage(hwnd, 0x7F, 1, 0);
if (iconHandle == IntPtr.Zero)
iconHandle = GetClassLongPtr(hwnd, -14);
if (iconHandle == IntPtr.Zero)
iconHandle = GetClassLongPtr(hwnd, -34);
if (iconHandle == IntPtr.Zero)
return null;
Icon icn = System.Drawing.Icon.FromHandle(iconHandle);
return icn;
}
public class WindowInformation {
public string Title { get; set; }
public Bitmap WindowBitmap { get; set; }
public Icon AppIcon { get; set; }
public bool IsVisible { get; set; }
public int Width { get; set; }
public int Height { get; set; }
public RECT Rect { get; set; }
public WINDOWPLACEMENT Placement { get; set; }
public HWND hwnd { get; set; }
public RECT RealRect { get; set; }
public Rectangle ContentRect { get; set; }
public IntPtr Handle { get; set; }
public int WindowDPI { get; set; }
public int SystemDPI { get; set; }
public double DPIScale { get; set; }
}
public struct WINDOWPLACEMENT {
public int length;
public int flags;
public int showCmd;
public System.Drawing.Point ptMinPosition;
public System.Drawing.Point ptMaxPosition;
public System.Drawing.Rectangle rcNormalPosition;
public static WINDOWPLACEMENT Default {
get {
WINDOWPLACEMENT result = new WINDOWPLACEMENT();
result.length = Marshal.SizeOf(result);
return result;
}
}
}
public delegate bool EnumDesktopWindowsDelegate(IntPtr hWnd, int lParam);
public WindowInformation[] GetAllWindows(HWND[] excludedHwnds) {
var windows = new List<WindowInformation>();
IntPtr hShellWnd = GetShellWindow();
IntPtr hDefView = FindWindowEx(hShellWnd, IntPtr.Zero, "SHELLDLL_DefView", null);
IntPtr folderView = FindWindowEx(hDefView, IntPtr.Zero, "SysListView32", null);
IntPtr taskBar = FindWindowEx(IntPtr.Zero, IntPtr.Zero, "Shell_TrayWnd", null);
var excluded = new List<HWND>() {
new HWND(hShellWnd), new HWND(hDefView), new HWND(folderView), new HWND(taskBar)
};
var excludedWindowTitle = new string[] {
"NVIDIA GeForce Overlay", "Ink Canvas 画板", "Ink Canvas Annotation", "Ink Canvas Artistry", "InkCanvasForClass"
};
excluded.AddRange(excludedHwnds);
if (!EnumDesktopWindows(IntPtr.Zero, new EnumDesktopWindowsDelegate((hwnd, param) => {
// 排除被過濾的窗體句柄
if (excluded.Contains(new HWND(hwnd))) return true;
// 判斷窗體是否可見
var isvisible = IsWindowVisible(hwnd);
if (!isvisible) return true;
// 判斷窗體透明度和額外樣式
var windowLong = (int)GetWindowLongPtr(hwnd, -20);
GetLayeredWindowAttributes(hwnd, out uint crKey, out byte bAlpha, out uint dwFlags);
if ((windowLong & 0x00000080L) != 0) return true;
if ((windowLong & 0x00080000) != 0 && (dwFlags & 0x00000002) != 0 && bAlpha == 0) return true; //分层窗口且全透明
// Win8+專用,用於檢測UWP應用是否隱藏
bool isCloacked = false;
if (OSVersion.GetOperatingSystem() > OperatingSystem.Windows7)
DwmGetWindowAttribute(hwnd, (int)DwmWindowAttribute.Cloaked, out isCloacked, Marshal.SizeOf(typeof(bool)));
if (isCloacked) return true;
// 獲取窗體實際大小
DwmGetWindowAttribute(hwnd, DwmWindowAttribute.ExtendedFrameBounds, out RECT realRect, Marshal.SizeOf(typeof(RECT)));
// 獲取窗體的進程ID
var pidRes = GetWindowThreadProcessId(hwnd, out uint pid);
if (pid == 0 || pidRes == 0) return true;
// 獲取窗體的DPI差異,scale為1則代表非DWM強制拉伸顯示窗體,實際截圖會根據窗體的DPI Awareness來截取
var dpiForHwnd = GetDpiForWindow(hwnd);
var dpiXProperty = typeof(SystemParameters).GetProperty("DpiX", BindingFlags.NonPublic | BindingFlags.Static);
var dpiYProperty = typeof(SystemParameters).GetProperty("Dpi", BindingFlags.NonPublic | BindingFlags.Static);
var dpiX = (int)dpiXProperty.GetValue(null, null);
var dpiY = (int)dpiYProperty.GetValue(null, null);
var dpi = (dpiX + dpiY) / 2;
double scale = 1;
if (dpi > dpiForHwnd) { // 说明该应用是win32应用,靠DWM的拉伸放大到高DPI
scale = dpi / (double)dpiForHwnd;
}
// 獲取窗體應用程式圖標
var icon = GetAppIcon(hwnd);
// 獲取應用程式標題,這裡空標題不略過,用於後續繼續判斷獲取標題
var length = GetWindowTextLength(hwnd) + 1;
var title = new StringBuilder(length);
GetWindowText(hwnd, title, length);
// if (title.ToString().Length == 0) return true;
// 窗體標題黑名單,在黑名單中的窗體不會顯示
if (excludedWindowTitle.Equals(title.ToString())) return true;
// 獲取窗體狀態,如果是最小化就跳過
WINDOWPLACEMENT placement = new WINDOWPLACEMENT();
GetWindowPlacement(hwnd, ref placement);
if (placement.showCmd == 2) return true;
// 獲取窗口Rect,用於和DwmGetWindowAttribute方法獲取到的窗體大小進行Offset計算
RECT rect;
GetWindowRect(hwnd, out rect);
var w = rect.Width;
var h = rect.Height;
if (w == 0 || h == 0) return true;
// 使用PrintWindowRENDER_FULL_CONTENT)來實現窗體圖片截取(支持D3D和DX)
Bitmap bmp = new Bitmap(rect.Width, rect.Height);
Graphics memoryGraphics = Graphics.FromImage(bmp);
IntPtr hdc = memoryGraphics.GetHdc();
PrintWindow(hwnd, hdc, 2);
// 添加窗體信息
windows.Add(new WindowInformation() {
AppIcon = icon,
Title = title.ToString(),
IsVisible = isvisible,
WindowBitmap = bmp,
Width = w,
Height = h,
Rect = rect,
Placement = placement,
RealRect = realRect,
Handle = hwnd,
ContentRect = new Rectangle(realRect.X - rect.X, realRect.Y - rect.Y, (int)Math.Round(
realRect.Width / scale ,0), (int)Math.Round(realRect.Height / scale, 0)),
WindowDPI = dpiForHwnd,
SystemDPI = dpi,
DPIScale = scale
});
// 釋放HDC
memoryGraphics.ReleaseHdc(hdc);
// 嘗試調用GC回收叻色
System.GC.Collect();
System.GC.WaitForPendingFinalizers();
return true;
}),
IntPtr.Zero)) return new WindowInformation[] { };
return windows.ToArray();
}
public static string GetProcessPathByPid(int processId) {
string query = $"SELECT Name, ExecutablePath FROM Win32_Process WHERE ProcessId = {processId}";
ManagementObjectSearcher searcher = new ManagementObjectSearcher(query);
foreach (ManagementObject obj in searcher.Get()) {
string executablePath = obj["ExecutablePath"]?.ToString();
if (!string.IsNullOrEmpty(executablePath)) return executablePath;
}
return "";
}
public async Task<string> GetProcessPathByPidAsync(int processId) {
var result = await Task.Run(() => GetProcessPathByPid(processId));
return result;
}
private static string GetAppFriendlyName(string filePath)
{
var versionInfo = FileVersionInfo.GetVersionInfo(filePath);
return versionInfo.FileDescription;
}
public async Task<WindowInformation[]> GetAllWindowsAsync(HWND[] excludedHwnds) {
var windows = await Task.Run(() => GetAllWindows(excludedHwnds));
var _wins = new List<WindowInformation>(){};
foreach (var w in windows) {
_wins.Add(w);
}
foreach (var windowInformation in windows) {
if (windowInformation.Title.Length == 0) {
GetWindowThreadProcessId(windowInformation.Handle, out uint Pid);
if (Pid != 0) {
var _path = Path.GetFullPath(await GetProcessPathByPidAsync((int)Pid));
var processPath = Path.GetFullPath(Process.GetCurrentProcess().MainModule.FileName);
if (string.Equals(_path, processPath, StringComparison.OrdinalIgnoreCase) || _path == "") {
_wins.Remove(windowInformation);
} else {
var _des = GetAppFriendlyName(_path);
Trace.WriteLine(_des);
if (_des == null) {
_wins.Remove(windowInformation);
} else {
var index = _wins.IndexOf(windowInformation);
_wins[index].Title = _des;
}
}
} else {
_wins.Remove(windowInformation);
}
}
}
return _wins.ToArray();
}
#endregion
#region
private Bitmap GetScreenshotBitmap() {
Rectangle rc = System.Windows.Forms.SystemInformation.VirtualScreen;
var bitmap = new Bitmap(rc.Width, rc.Height, PixelFormat.Format32bppArgb);
using (Graphics memoryGrahics = Graphics.FromImage(bitmap)) {
memoryGrahics.CopyFromScreen(rc.X, rc.Y, 0, 0, rc.Size, CopyPixelOperation.SourceCopy);
}
return bitmap;
}
#endregion
#region API
private BitmapImage BitmapToImageSource(Bitmap bitmap) {
using (MemoryStream memory = new MemoryStream()) {
bitmap.Save(memory, System.Drawing.Imaging.ImageFormat.Bmp);
memory.Position = 0;
BitmapImage bitmapimage = new BitmapImage();
bitmapimage.BeginInit();
bitmapimage.StreamSource = memory;
bitmapimage.CacheOption = BitmapCacheOption.OnLoad;
bitmapimage.EndInit();
return bitmapimage;
}
}
public enum SnapshotMethod {
Auto,
GraphicsAPICopyFromScreen,
MagnificationAPIWithPrintWindow,
MagnificationAPIWithCallback
}
public enum OutputImageMIMEFormat {
Png,
Bmp,
Jpeg,
}
public class SnapshotConfig {
public SnapshotMethod SnapshotMethod { get; set; } = SnapshotMethod.Auto;
public bool IsCopyToClipboard { get; set; } = false;
public bool IsSaveToLocal { get; set; } = true;
public DirectoryInfo BitmapSavePath { get; set; } = null;
public string SaveBitmapFileName { get; set; } = "Screenshot-[YYYY]-[MM]-[DD]-[HH]-[mm]-[ss].png";
public OutputImageMIMEFormat OutputMIMEType { get; set; } = OutputImageMIMEFormat.Png;
public HWND[] ExcludedHwnds { get; set; } = new HWND[] { };
}
private static ImageCodecInfo GetEncoderInfo(string mimeType) {
foreach (ImageCodecInfo codec in ImageCodecInfo.GetImageEncoders())
if (codec.MimeType == mimeType)
return codec;
return null;
}
public async Task<Bitmap> FullscreenSnapshot(SnapshotConfig config) {
Bitmap bitmap = new Bitmap(1, 1);
var ex = new List<HWND>() { new HWND(new WindowInteropHelper(this).Handle) };
ex.AddRange(config.ExcludedHwnds);
if (config.SnapshotMethod == SnapshotMethod.Auto) {
if (OSVersion.GetOperatingSystem() >= OperatingSystem.Windows81) {
bitmap = await SaveScreenshotToDesktopByMagnificationAPIAsync(ex.ToArray(), false);
} else {
if (ex.Count != 0)
foreach (var hwnd in ex)
ShowWindow(hwnd.DangerousGetHandle(), 0);
bitmap = GetScreenshotBitmap();
foreach (var hwnd in ex) ShowWindow(hwnd.DangerousGetHandle(), 5);
}
} else if (config.SnapshotMethod == SnapshotMethod.MagnificationAPIWithPrintWindow ||
config.SnapshotMethod == SnapshotMethod.MagnificationAPIWithCallback) {
if (!(OSVersion.GetOperatingSystem() >= OperatingSystem.Windows81))
throw new Exception("您的系統版本不支持 MagnificationAPI 截圖!");
bitmap = await SaveScreenshotToDesktopByMagnificationAPIAsync(ex.ToArray(),
config.SnapshotMethod == SnapshotMethod.MagnificationAPIWithCallback);
} else if (config.SnapshotMethod == SnapshotMethod.GraphicsAPICopyFromScreen) {
if (ex.Count != 0)
foreach (var hwnd in ex)
ShowWindow(hwnd.DangerousGetHandle(), 0);
bitmap = GetScreenshotBitmap();
foreach (var hwnd in ex) ShowWindow(hwnd.DangerousGetHandle(), 5);
}
if (bitmap.Width == 1 && bitmap.Height == 1) throw new Exception("截圖失敗");
try {
if (config.IsCopyToClipboard) Clipboard.SetImage(BitmapToImageSource(bitmap));
}
catch (NotSupportedException e) { }
if (config.IsSaveToLocal) {
var fullPath = config.BitmapSavePath.FullName;
if (!config.BitmapSavePath.Exists) config.BitmapSavePath.Create();
var fileName = config.SaveBitmapFileName.Replace("[YYYY]", DateTime.Now.Year.ToString())
.Replace("[MM]", DateTime.Now.Month.ToString()).Replace("[DD]", DateTime.Now.Day.ToString())
.Replace("[HH]", DateTime.Now.Hour.ToString()).Replace("[mm]", DateTime.Now.Minute.ToString())
.Replace("[ss]", DateTime.Now.Second.ToString()).Replace("[width]", bitmap.Width.ToString())
.Replace("[height]", bitmap.Height.ToString());
var finalPath = (fullPath.EndsWith("\\") ? fullPath.Substring(0, fullPath.Length - 1) : fullPath) +
$"\\{fileName}";
bitmap.Save(finalPath, config.OutputMIMEType == OutputImageMIMEFormat.Png ? ImageFormat.Png :
config.OutputMIMEType == OutputImageMIMEFormat.Bmp ? ImageFormat.Bmp : ImageFormat.Jpeg);
}
bitmap.Dispose();
return bitmap;
}
#endregion
private void SaveScreenshot(bool isHideNotification, string fileName = null) {
var bitmap = GetScreenshotBitmap();
string savePath = Settings.Automation.AutoSavedStrokesLocation + @"\Auto Saved - Screenshots";
if (fileName == null) fileName = DateTime.Now.ToString("u").Replace(":", "-");
if (Settings.Automation.IsSaveScreenshotsInDateFolders) {
savePath += @"\" + DateTime.Now.ToString("yyyy-MM-dd");
}
savePath += @"\" + fileName + ".png";
if (!Directory.Exists(Path.GetDirectoryName(savePath))) {
Directory.CreateDirectory(Path.GetDirectoryName(savePath));
}
bitmap.Save(savePath, ImageFormat.Png);
if (Settings.Automation.IsAutoSaveStrokesAtScreenshot) {
SaveInkCanvasStrokes(false, false);
}
if (!isHideNotification) {
ShowNewToast("截图成功保存至 " + savePath, MW_Toast.ToastType.Success, 3000);
}
}
private void SaveScreenShotToDesktop() {
var bitmap = GetScreenshotBitmap();
string savePath = Environment.GetFolderPath(Environment.SpecialFolder.DesktopDirectory);
bitmap.Save(savePath + @"\" + DateTime.Now.ToString("u").Replace(':', '-') + ".png", ImageFormat.Png);
ShowNewToast("截图成功保存至【桌面" + @"\" + DateTime.Now.ToString("u").Replace(':', '-') + ".png】",
MW_Toast.ToastType.Success, 3000);
if (Settings.Automation.IsAutoSaveStrokesAtScreenshot) SaveInkCanvasStrokes(false, false);
}
private void SavePPTScreenshot(string fileName) {
var bitmap = GetScreenshotBitmap();
string savePath = Settings.Automation.AutoSavedStrokesLocation + @"\Auto Saved - PPT Screenshots";
if (Settings.Automation.IsSaveScreenshotsInDateFolders) {
savePath += @"\" + DateTime.Now.ToString("yyyy-MM-dd");
}
if (fileName == null) fileName = DateTime.Now.ToString("u").Replace(":", "-");
savePath += @"\" + fileName + ".png";
if (!Directory.Exists(Path.GetDirectoryName(savePath))) {
Directory.CreateDirectory(Path.GetDirectoryName(savePath));
}
bitmap.Save(savePath, ImageFormat.Png);
if (Settings.Automation.IsAutoSaveStrokesAtScreenshot) {
SaveInkCanvasStrokes(false, false);
}
}
}
}