获取特定应用程序的屏幕截图

Hao*_*Lim 48 c# winapi screenshot

我知道我可以使用Graphics.CopyFromScreen()获取整个屏幕的屏幕截图.但是,如果我只想要特定应用程序的屏幕截图怎么办?

Mau*_*gan 105

PrintWindow win32 api将捕获窗口位图,即使窗口被其他窗口覆盖或者它是否在屏幕外:

[DllImport("user32.dll")]
public static extern bool GetWindowRect(IntPtr hWnd, out RECT lpRect);
[DllImport("user32.dll")]
public static extern bool PrintWindow(IntPtr hWnd, IntPtr hdcBlt, int nFlags);

public static Bitmap PrintWindow(IntPtr hwnd)    
{       
    RECT rc;        
    GetWindowRect(hwnd, out rc);

    Bitmap bmp = new Bitmap(rc.Width, rc.Height, PixelFormat.Format32bppArgb);        
    Graphics gfxBmp = Graphics.FromImage(bmp);        
    IntPtr hdcBitmap = gfxBmp.GetHdc();        

    PrintWindow(hwnd, hdcBitmap, 0);  

    gfxBmp.ReleaseHdc(hdcBitmap);               
    gfxBmp.Dispose(); 

    return bmp;   
}
Run Code Online (Sandbox Code Playgroud)

上面对RECT的引用可以通过以下类解决:

[StructLayout(LayoutKind.Sequential)]
public struct RECT
{
    private int _Left;
    private int _Top;
    private int _Right;
    private int _Bottom;

    public RECT(RECT Rectangle) : this(Rectangle.Left, Rectangle.Top, Rectangle.Right, Rectangle.Bottom)
    {
    }
    public RECT(int Left, int Top, int Right, int Bottom)
    {
        _Left = Left;
        _Top = Top;
        _Right = Right;
        _Bottom = Bottom;
    }

    public int X {
        get { return _Left; }
        set { _Left = value; }
    }
    public int Y {
        get { return _Top; }
        set { _Top = value; }
    }
    public int Left {
        get { return _Left; }
        set { _Left = value; }
    }
    public int Top {
        get { return _Top; }
        set { _Top = value; }
    }
    public int Right {
        get { return _Right; }
        set { _Right = value; }
    }
    public int Bottom {
        get { return _Bottom; }
        set { _Bottom = value; }
    }
    public int Height {
        get { return _Bottom - _Top; }
        set { _Bottom = value + _Top; }
    }
    public int Width {
        get { return _Right - _Left; }
        set { _Right = value + _Left; }
    }
    public Point Location {
        get { return new Point(Left, Top); }
        set {
            _Left = value.X;
            _Top = value.Y;
        }
    }
    public Size Size {
        get { return new Size(Width, Height); }
        set {
            _Right = value.Width + _Left;
            _Bottom = value.Height + _Top;
        }
    }

    public static implicit operator Rectangle(RECT Rectangle)
    {
        return new Rectangle(Rectangle.Left, Rectangle.Top, Rectangle.Width, Rectangle.Height);
    }
    public static implicit operator RECT(Rectangle Rectangle)
    {
        return new RECT(Rectangle.Left, Rectangle.Top, Rectangle.Right, Rectangle.Bottom);
    }
    public static bool operator ==(RECT Rectangle1, RECT Rectangle2)
    {
        return Rectangle1.Equals(Rectangle2);
    }
    public static bool operator !=(RECT Rectangle1, RECT Rectangle2)
    {
        return !Rectangle1.Equals(Rectangle2);
    }

    public override string ToString()
    {
        return "{Left: " + _Left + "; " + "Top: " + _Top + "; Right: " + _Right + "; Bottom: " + _Bottom + "}";
    }

    public override int GetHashCode()
    {
        return ToString().GetHashCode();
    }

    public bool Equals(RECT Rectangle)
    {
        return Rectangle.Left == _Left && Rectangle.Top == _Top && Rectangle.Right == _Right && Rectangle.Bottom == _Bottom;
    }

    public override bool Equals(object Object)
    {
        if (Object is RECT) {
            return Equals((RECT)Object);
        } else if (Object is Rectangle) {
            return Equals(new RECT((Rectangle)Object));
        }

        return false;
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 好的解决方案 我只是想指出,有时PixelFormat.Format32bppArgb会产生白色瑕疵.在这种情况下,只需尝试使用其他格式,例如PixelFormat.Format24bppRgb (3认同)
  • 我该如何使用这种方法?PrintWindow(`这里要传递什么?) (3认同)
  • @MauriceFlanagan 我得到了全黑的图像?为什么? (3认同)
  • 黑色图像解决方案是否已添加到此解决方案中?我在Windows 10上遇到了同样的问题 (3认同)
  • UWP 应用程序始终为黑色位图。当我与 UWP 应用程序位于同一桌面上时,我什至似乎无法获取 UWP 应用程序的 DWM 缩略图。但是,当我不在同一桌面上时,我可以获取 UWP 应用程序的 DWM 缩略图。这让我疯狂。 (2认同)
  • 我找到了解决黑色图像问题的方法。尝试将值 2 作为“PrintWindow”的第三个参数(标志)。经过一番挖掘,我在 chromium 源代码中发现了以下内容:_“PW_RENDERFULLCONTENT 标志未记录,但从 Windows 8.1 开始工作。它允许捕获使用 DirectComposition 绘制的窗口的内容。”_`UINT flags = PW_CLIENTONLY | PW_RENDERFULLCONTENT;` 所以我这样做:`int PW_CLIENTONLY = 0x1; int PW_RENDERFULLCONTENT = 0x2;PrintWindow(hwnd, hdcBitmap, PW_CLIENTONLY | PW_RENDERFULLCONTENT);` (2认同)

Alc*_*nja 35

这里有一些代码可以帮助您入门:

public void CaptureApplication(string procName)
{
    var proc = Process.GetProcessesByName(procName)[0];
    var rect = new User32.Rect();
    User32.GetWindowRect(proc.MainWindowHandle, ref rect);

    int width = rect.right - rect.left;
    int height = rect.bottom - rect.top;

    var bmp = new Bitmap(width, height, PixelFormat.Format32bppArgb);
    Graphics graphics = Graphics.FromImage(bmp);
    graphics.CopyFromScreen(rect.left, rect.top, 0, 0, new Size(width, height), CopyPixelOperation.SourceCopy);

    bmp.Save("c:\\tmp\\test.png", ImageFormat.Png);
}

private class User32
{
    [StructLayout(LayoutKind.Sequential)]
    public struct Rect
    {
        public int left;
        public int top;
        public int right;
        public int bottom;
    }

    [DllImport("user32.dll")]
    public static extern IntPtr GetWindowRect(IntPtr hWnd, ref Rect rect);
}
Run Code Online (Sandbox Code Playgroud)

它有效,但需要改进:

  • 您可能希望使用不同的机制来获取进程句柄(或者至少进行一些防御性编码)
  • 如果您的目标窗口不在前景中,您最终会得到一个正确大小/位置的屏幕截图,但只会填充前景中的任何内容(您可能希望先将给定窗口拉到前台) )
  • 除了将bmp保存到临时目录之外,您可能还想做其他事情

  • @alconja我用您的代码拍摄了记事本的快照。但是它拍摄了Visual Studio的快照,这是活动窗口。我们可以用它拍摄没有活动窗口的照片吗? (2认同)

Hao*_*Lim 9

根据Alconja的回答,我做了一些改进:

[StructLayout(LayoutKind.Sequential)]
public struct Rect
{
    public int left;
    public int top;
    public int right;
    public int bottom;
}

[DllImport("user32.dll")]
private static extern int SetForegroundWindow(IntPtr hWnd);

private const int SW_RESTORE = 9;

[DllImport("user32.dll")]
private static extern IntPtr ShowWindow(IntPtr hWnd, int nCmdShow);

[DllImport("user32.dll")]
public static extern IntPtr GetWindowRect(IntPtr hWnd, ref Rect rect);

public Bitmap CaptureApplication(string procName)
{
    Process proc;

    // Cater for cases when the process can't be located.
    try
    {
        proc = Process.GetProcessesByName(procName)[0];
    }
    catch (IndexOutOfRangeException e)
    {
        return null;
    }

    // You need to focus on the application
    SetForegroundWindow(proc.MainWindowHandle);
    ShowWindow(proc.MainWindowHandle, SW_RESTORE);

    // You need some amount of delay, but 1 second may be overkill
    Thread.Sleep(1000);

    Rect rect = new Rect();
    IntPtr error = GetWindowRect(proc.MainWindowHandle, ref rect);

    // sometimes it gives error.
    while (error == (IntPtr)0)
    {
        error = GetWindowRect(proc.MainWindowHandle, ref rect);
    }

    int width = rect.right - rect.left;
    int height = rect.bottom - rect.top;

    Bitmap bmp = new Bitmap(width, height, PixelFormat.Format32bppArgb);
    Graphics.FromImage(bmp).CopyFromScreen(rect.left,
                                           rect.top,
                                           0,
                                           0,
                                           new Size(width, height),
                                           CopyPixelOperation.SourceCopy);

    return bmp;
}
Run Code Online (Sandbox Code Playgroud)

  • 浏览完之后我注意到,如果进程在`Thread.Sleep(1000)期间关闭,`你将有一个无限循环. (3认同)
  • 这个答案也没有处理“ Graphics”对象。 (3认同)