C# WebBrowser.DrawToBitmap()或其他方法?
我正在尝试捕获WebBrowser控件的内容C# WebBrowser.DrawToBitmap()或其他方法?,c#,.net,html,vb.net,webbrowser-control,C#,.net,Html,Vb.net,Webbrowser Control,我正在尝试捕获WebBrowser控件的内容DrawToBitmap()可以很好地工作,但WebBrowser控件的文档中不支持它。我一直在试图找到另一种方法来捕获WebBrowser控件的内容并将其保存到本地图像文件中 是否有人有任何变通方法或其他方法将WebBrowser控件的内容保存到本地图像文件?控件.DrawToBitmap并不总是有效,因此我求助于以下本机API调用,以提供更一致的结果: 实用程序类。调用Utilities.CaptureWindow(Control.Handle)以
DrawToBitmap()
可以很好地工作,但WebBrowser控件的文档中不支持它。我一直在试图找到另一种方法来捕获WebBrowser控件的内容并将其保存到本地图像文件中
是否有人有任何变通方法或其他方法将WebBrowser控件的内容保存到本地图像文件?控件.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);
}
}
这是我的项目中的一个代码片段,我专门编写它来从WebBrowser控件截图。我正在使用DrawToBitmap(Visual Studio 2008 C#)来捕获大图像(用户签名的发票、屏幕外的内容)。基本上,它工作得很好,但我得到的是空白图像。大约有100名员工在使用这个软件,大约每两周我就能看到一张空白图像
我做了很多测试,发现一件有趣的事情是:
我创建了一个按钮来从webbrowser生成图像。通常是可以的,但是如果我先点击Web浏览器,然后点击创建按钮,就会出现空白图像。 < P>即使窗口大于屏幕的大小,下面的方法也可以捕获整个窗口图像。然后,如果将窗口调整为
webBrowser.Document.offsetlectangle.Size,它就可以捕获页面内容的图像
class NativeMethods
{
[ComImport]
[Guid("0000010D-0000-0000-C000-000000000046")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
interface IViewObject
{
void Draw([MarshalAs(UnmanagedType.U4)] uint dwAspect, int lindex, IntPtr pvAspect, [In] IntPtr ptd, IntPtr hdcTargetDev, IntPtr hdcDraw, [MarshalAs(UnmanagedType.Struct)] ref RECT lprcBounds, [In] IntPtr lprcWBounds, IntPtr pfnContinue, [MarshalAs(UnmanagedType.U4)] uint dwContinue);
}
[StructLayout(LayoutKind.Sequential, Pack = 4)]
struct RECT
{
public int Left;
public int Top;
public int Right;
public int Bottom;
}
public static void GetImage(object obj, Image destination, Color backgroundColor)
{
using(Graphics graphics = Graphics.FromImage(destination))
{
IntPtr deviceContextHandle = IntPtr.Zero;
RECT rectangle = new RECT();
rectangle.Right = destination.Width;
rectangle.Bottom = destination.Height;
graphics.Clear(backgroundColor);
try
{
deviceContextHandle = graphics.GetHdc();
IViewObject viewObject = obj as IViewObject;
viewObject.Draw(1, -1, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, deviceContextHandle, ref rectangle, IntPtr.Zero, IntPtr.Zero, 0);
}
finally
{
if(deviceContextHandle != IntPtr.Zero)
{
graphics.ReleaseHdc(deviceContextHandle);
}
}
}
}
}
用法:
Bitmap screenshot = new Bitmap(1024, 768);
NativeMethods.GetImage(webBrowser.ActiveXInstance, screenshot, Color.White);
我使用了OleDraw方法,就像在SO中一样,但将其集成到了一个从WebBrowser派生的类中。这不仅允许对WebBrowser执行普通Control.DrawToBitmap,还允许对包含WebBrowser的表单执行普通Control.DrawToBitmap。如果表单是隐藏的(被另一个表单覆盖,包括MDI父表单),并且当用户锁定了与Win+L的会话(我还没有测试它)时,这也可以工作
谢谢,我会试试看。@Matt:希望有帮助。我在最初的帖子中漏掉了Gdi32类,但现在已经修复了。@Cory Charlton:它工作得很好!我唯一的问题是假设WebBrowser控件中有一个扩展到多个页面的页面。没有办法用这个来抓住它?@Matt:我相信这只会得到控件的可见部分。我还没有尝试过,但是您可以尝试使控件足够大,以容纳整个内容,但不一定在表单上可见。我希望在这种情况下您会得到一个空图像,但值得一试。@Cory我尝试过这种方法,不幸的是它似乎不起作用。它生成的所有截图只是黑箱,包括用于标准的DoptoBMP映射调用的URL。微软自己关于这个方法的文档告诉人们不要使用它,但每个人都会这样做,因为这是一个非常需要的方法。我一直在寻找解决这个问题的办法,已经有一个星期了。我必须依靠IE进行渲染,因此无法使用其他解决方案。看看是否有更好的方法从DrawToBitmap方法中获得一致的结果是很有趣的。只是一个建议,如果我无法解决它,我可能必须自己做一些事情,否则,你应该能够确定图像是空白的,不仅每个像素都是白色的,但是,如果它们都是白色的,那么一点数学应该表明图像中只有一个颜色像素。相关的q/a:-最后的解决方案很简单,似乎是可靠的。如果html页面有卷轴,上面能够截图。我试着使用@Cory Charlton它确实有效,但只在可见屏幕上有效,而不是整个页面
public class WebBrowserEx : WebBrowser
{
private const uint DVASPECT_CONTENT = 1;
[DllImport("ole32.dll", PreserveSig = false)]
private static extern void OleDraw([MarshalAs(UnmanagedType.IUnknown)] object pUnk,
uint dwAspect,
IntPtr hdcDraw,
[In] ref System.Drawing.Rectangle lprcBounds
);
protected override void WndProc(ref Message m)
{
const int WM_PRINT = 0x0317;
switch (m.Msg)
{
case WM_PRINT:
Rectangle browserRect = new Rectangle(0, 0, this.Width, this.Height);
// Don't know why, but drawing with OleDraw directly on HDC from m.WParam.
// results in badly scaled (stretched) image of the browser.
// So, drawing to an intermediate bitmap first.
using (Bitmap browserBitmap = new Bitmap(browserRect.Width, browserRect.Height))
{
using (var graphics = Graphics.FromImage(browserBitmap))
{
var hdc = graphics.GetHdc();
OleDraw(this.ActiveXInstance, DVASPECT_CONTENT, hdc, ref browserRect);
graphics.ReleaseHdc(hdc);
}
using (var graphics = Graphics.FromHdc(m.WParam))
{
graphics.DrawImage(browserBitmap, Point.Empty);
}
}
// ignore default WndProc
return;
}
base.WndProc(ref m);
}
}