Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/csharp/339.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
使用C#、WPF和DWM保存窗口的屏幕截图_C#_Wpf_Screenshot_Dwm - Fatal编程技术网

使用C#、WPF和DWM保存窗口的屏幕截图

使用C#、WPF和DWM保存窗口的屏幕截图,c#,wpf,screenshot,dwm,C#,Wpf,Screenshot,Dwm,这是我们的后续问题 上述解决方案使用DWM显示活动窗口的缩略图。如果我理解正确,它的工作原理是让您指定要查看的应用程序的窗口句柄,然后让您提供窗口句柄和窗口上的一个位置,在该窗口上,窗口应该在其中绘制目标窗口的内容 有没有办法将窗口屏幕快照直接渲染到BitmapImage或Image,而不是直接在窗口中的某个位置绘制?(基本上只需抓取窗口的屏幕截图,即使它被另一个窗口覆盖,也不需要使用更新的缩略图。) 谢谢你的帮助 使用 Control.DrawToBitmap并不总是有效,因此我求助于以下本机

这是我们的后续问题

上述解决方案使用DWM显示活动窗口的缩略图。如果我理解正确,它的工作原理是让您指定要查看的应用程序的窗口句柄,然后让您提供窗口句柄和窗口上的一个位置,在该窗口上,窗口应该在其中绘制目标窗口的内容

有没有办法将窗口屏幕快照直接渲染到BitmapImage或Image,而不是直接在窗口中的某个位置绘制?(基本上只需抓取窗口的屏幕截图,即使它被另一个窗口覆盖,也不需要使用更新的缩略图。)

谢谢你的帮助

使用


Control.DrawToBitmap并不总是有效,因此我求助于以下本机API调用,以提供更一致的结果:

实用程序类。调用Utilities.CaptureWindow(Control.Handle)以捕获特定控件:

public static class Utilities
{
    public static Image CaptureScreen()
    {
        return CaptureWindow(User32.GetDesktopWindow());
    }

    public static Image CaptureWindow(IntPtr handle)
    {

        IntPtr hdcSrc = User32.GetWindowDC(handle);

        RECT windowRect = new RECT();
        User32.GetWindowRect(handle, ref windowRect);

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

        IntPtr hdcDest = Gdi32.CreateCompatibleDC(hdcSrc);
        IntPtr hBitmap = Gdi32.CreateCompatibleBitmap(hdcSrc, width, height);

        IntPtr hOld = Gdi32.SelectObject(hdcDest, hBitmap);
        Gdi32.BitBlt(hdcDest, 0, 0, width, height, hdcSrc, 0, 0, ApiConstants.SRCCOPY);
        Gdi32.SelectObject(hdcDest, hOld);
        Gdi32.DeleteDC(hdcDest);
        User32.ReleaseDC(handle, hdcSrc);

        Image image = Image.FromHbitmap(hBitmap);
        Gdi32.DeleteObject(hBitmap);

        return image;
    }
}
Gdi32类:

public class Gdi32
{
    [DllImport("gdi32.dll")]
    public static extern bool BitBlt(IntPtr hObject, int nXDest, int nYDest, int nWidth, int nHeight, IntPtr hObjectSource, int nXSrc, int nYSrc, int dwRop);
    [DllImport("gdi32.dll")]
    public static extern IntPtr CreateCompatibleBitmap(IntPtr hDC, int nWidth, int nHeight);
    [DllImport("gdi32.dll")]
    public static extern IntPtr CreateCompatibleDC(IntPtr hDC);
    [DllImport("gdi32.dll")]
    public static extern bool DeleteDC(IntPtr hDC);
    [DllImport("gdi32.dll")]
    public static extern bool DeleteObject(IntPtr hObject);
    [DllImport("gdi32.dll")]
    public static extern IntPtr SelectObject(IntPtr hDC, IntPtr hObject);
}
User32类:

public static class User32
{
    [DllImport("user32.dll")]
    public static extern IntPtr GetDesktopWindow();
    [DllImport("user32.dll")]
    public static extern IntPtr GetWindowDC(IntPtr hWnd);
    [DllImport("user32.dll")]
    public static extern IntPtr GetWindowRect(IntPtr hWnd, ref RECT rect);
    [DllImport("user32.dll")]
    public static extern IntPtr ReleaseDC(IntPtr hWnd, IntPtr hDC);
}
使用的常数:

    public const int SRCCOPY = 13369376;
使用的结构:

[StructLayout(LayoutKind.Sequential)]
public struct RECT
{
    public int left;
    public int top;
    public int right;
    public int bottom;
}
友好的控件扩展方法:

public static class ControlExtensions
{
    public static Image DrawToImage(this Control control)
    {
        return Utilities.CaptureWindow(control.Handle);
    }
}

谢谢你的回复。这不需要先在屏幕上绘制屏幕快照,然后保存吗?有没有一种方法可以直接将窗口的屏幕截图捕捉到对象(如图像),而不必先在屏幕上绘制它?DrawToBitmap会像按PrintScreen按钮(在窗体上或仅用于面板)时一样获取位图。我不确定窗体或面板是否应该真正显示在屏幕上,使用可见面板肯定有效,我相信使用隐藏面板也应该有效。位图bm=新位图(wid,hgt);'将当前表单绘制到位图this.DrawToBitmap(bm,新矩形(0,0,wid,hgt))这很有用,但我在这里发布了一个后续问题:太好了,这是一个超快速的代码。但wpf控件没有句柄:(
public static class ControlExtensions
{
    public static Image DrawToImage(this Control control)
    {
        return Utilities.CaptureWindow(control.Handle);
    }
}