捕获网页为图像

参考原文(英文) http://www.codeproject.com/cs/media/IECapture.asp
环境:Visual Studio.NET 2003 语言:C#
系统需求:Windows + iexplore                                                       附件:捕获网页为图像程序
捕获效果图:

         该程序的目标很明确,就是捕获IE浏览器打开的网页存为图像,附加的需求,我们能自定义图像的保存质量(控制图像的大小)。当然我们还要考虑当有多个浏览器窗口打开的时候,我们是捕获所有窗口的图像。
        废话不多说,我们开始说说这个程序如何实现。

第一步:
         该程序需要SHDocVw.dll 和 MSHTML.dll的支持,所以在我们的工程中需要添加两个com组件,在添加引用的对话框中选择com的标签页,然后找到Microsoft Internet Controls". 和Microsoft HTML Object Library,添加进来。

第二步:添加必要的名称空间     

None.gif using  System.Text;
None.gif
using  System.Runtime.InteropServices;
None.gif
using  System.Diagnostics;
None.gif
using  System.IO;
None.gif
using  System.Drawing.Imaging;
None.gif
using  SHDocVw;
None.gif
using  mshtml;

第三步:调用user32.dll的函数
None.gif [DllImport( " user32.dll " , CharSet = CharSet.Auto)]
ExpandedBlockStart.gifContractedBlock.gif
public   static   extern  IntPtr FindWindowEx(IntPtr parent  /**/ /* HWND */
ExpandedBlockStart.gifContractedBlock.gif  IntPtr next 
/**/ /* HWND */ string  sClassName, IntPtr sWindowTitle);
None.gif
None.gif[DllImport(
" user32.dll " , ExactSpelling = true , CharSet = CharSet.Auto)] 
None.gif
public   static   extern  IntPtr GetWindow(IntPtr hWnd,  int  uCmd); 
None.gif
None.gif[DllImport(
" user32.Dll " )]
None.gif
public   static   extern   void  GetClassName( int  h, StringBuilder s,  int  nMaxCount);
None.gif
None.gif[DllImport(
" user32.dll " )]
None.gif
private   static   extern   bool  PrintWindow(IntPtr hwnd, IntPtr hdcBlt,  uint  nFlags);
None.gif
None.gif
public   const   int  GW_CHILD  =   5
None.gif
public   const   int  GW_HWNDNEXT  =   2 ;

第四步:找到一个打开的浏览器进程,并分配一个Browser Document给它。

None.gif SHDocVw.WebBrowser m_browser  =   null ;
None.gif SHDocVw.ShellWindows shellWindows 
=   new  SHDocVw.ShellWindowsClass();
None.gif 
None.gif 
// Find first availble browser window.
None.gif 
// Application can easily be modified to loop through and 
None.gif 
// capture all open windows.
None.gif
  string  filename;
None.gif  
foreach  (SHDocVw.WebBrowser ie  in  shellWindows)
ExpandedBlockStart.gifContractedBlock.gif  
dot.gif {
InBlock.gif      filename 
=  Path.GetFileNameWithoutExtension(ie.FullName).ToLower();
InBlock.gif      
if  (filename.Equals( " iexplore " ))
ExpandedSubBlockStart.gifContractedSubBlock.gif      
dot.gif {
InBlock.gif          m_browser 
=  ie;
InBlock.gif          
break ;  
ExpandedSubBlockEnd.gif      }

ExpandedBlockEnd.gif  }

None.gif  
if  (m_browser  ==   null )
ExpandedBlockStart.gifContractedBlock.gif  
dot.gif {   
InBlock.gif      MessageBox.Show(
" No Browser Open " );
InBlock.gif      
return ;
ExpandedBlockEnd.gif  }

None.gif
None.gif  
// Assign Browser Document
None.gif
  mshtml.IHTMLDocument2 myDoc  =  (mshtml.IHTMLDocument2)m_browser.Document;

第五步:获取屏幕和网页的高度和宽度
None.gif   // Set scrolling on.
None.gif
 myDoc.body.setAttribute( " scroll " " yes " 0 );
None.gif 
None.gif 
// Get Browser Window Height
None.gif
  int  heightsize  =  ( int )myDoc.body.getAttribute( " scrollHeight " 0 );
None.gif 
int  widthsize  =  ( int )myDoc.body.getAttribute( " scrollWidth " 0 );
None.gif 
None.gif 
// Get Screen Height
None.gif
  int  screenHeight  =  ( int )myDoc.body.getAttribute( " clientHeight " 0 );
None.gif 
int  screenWidth  =  ( int )myDoc.body.getAttribute( " clientWidth " 0 );

第六步:捕获算法
         这里要说明的是,其实没有什么组件可以让我们直接把网页完全捕获过来,我们还是从基本的一页一页捕捉,然后组合整个图像的方法来完成的。
 其基本思想是这样的:首先捕获该网页的第一个片段,然后滚动条滑道下一页,继续捕获下一个片段,然后这个片段组合到一个目标位图中,就像缝合起来。这个过程会循环进行,直到最后一个片段捕获完毕,缝合到目标位图中去。
         当然我们还可能遇到这样一种情况,就是网页比屏幕要宽,这是我们的程序仍然是先捕获第一个片段,然后浏览器滚动条水平移动,捕获剩余的部分,然后缝接起来,然后再进行上述的步骤。
None.gif // Get bitmap to hold screen fragment.
None.gif
 Bitmap bm  =   new  Bitmap(screenWidth, screenHeight, 
None.gif    System.Drawing.Imaging.PixelFormat.Format16bppRgb555);
None.gif 
None.gif 
// Create a target bitmap to draw into.
None.gif
 Bitmap bm2  =   new  Bitmap(widthsize  +  URLExtraLeft, heightsize  +  
None.gif    URLExtraHeight 
-  trimHeight, 
None.gif         System.Drawing.Imaging.PixelFormat.Format16bppRgb555);
None.gif Graphics g2 
=  Graphics.FromImage(bm2);
None.gif 
None.gif Graphics g 
=   null ;
None.gif IntPtr hdc;
None.gif Image screenfrag 
=   null ;
None.gif 
int  brwTop  =   0 ;
None.gif 
int  brwLeft  =   0 ;
None.gif 
int  myPage  =   0 ;
None.gif IntPtr myIntptr 
=  (IntPtr)m_browser.HWND;
None.gif 
None.gif 
// Get inner browser window.
None.gif
  int  hwndInt  =  myIntptr.ToInt32();
None.gif IntPtr hwnd 
=  myIntptr;
None.gif hwnd 
=  GetWindow(hwnd, GW_CHILD); 
None.gif StringBuilder sbc 
=   new  StringBuilder( 256 );
None.gif 
None.gif 
// Get Browser "Document" Handle
None.gif
  while  (hwndInt  !=   0
ExpandedBlockStart.gifContractedBlock.gif 
dot.gif
InBlock.gif     hwndInt 
= hwnd.ToInt32();
InBlock.gif     GetClassName(hwndInt, sbc, 
256);
InBlock.gif 
InBlock.gif     
if(sbc.ToString().IndexOf("Shell DocObject View"0> -1)
ExpandedSubBlockStart.gifContractedSubBlock.gif     
dot.gif{
InBlock.gif         hwnd 
= FindWindowEx(hwnd, IntPtr.Zero, 
InBlock.gif             
"Internet Explorer_Server", IntPtr.Zero);
InBlock.gif         
break;
ExpandedSubBlockEnd.gif     }
                
InBlock.gif     hwnd 
= GetWindow(hwnd, GW_HWNDNEXT);
ExpandedBlockEnd.gif  }
 
None.gif 
None.gif 
// Get Screen Height (for bottom up screen drawing)
None.gif
  while  ((myPage  *  screenHeight)  <  heightsize)
ExpandedBlockStart.gifContractedBlock.gif 
dot.gif {
InBlock.gif     myDoc.body.setAttribute(
"scrollTop", (screenHeight - 5* myPage, 0);
InBlock.gif     
++myPage;
ExpandedBlockEnd.gif }

None.gif 
None.gif 
// Rollback the page count by one
None.gif
  -- myPage;
None.gif 
None.gif 
int  myPageWidth  =   0 ;
None.gif  
while  ((myPageWidth  *  screenWidth)  <  widthsize)
ExpandedBlockStart.gifContractedBlock.gif 
dot.gif {
InBlock.gif     myDoc.body.setAttribute(
"scrollLeft", (screenWidth - 5* myPageWidth, 0);
InBlock.gif     brwLeft 
= (int)myDoc.body.getAttribute("scrollLeft"0);
InBlock.gif     
for (int i = myPage; i >= 0--i)
ExpandedSubBlockStart.gifContractedSubBlock.gif     
dot.gif{
InBlock.gif         
//Shoot visible window
InBlock.gif
         g = Graphics.FromImage(bm);
InBlock.gif         hdc 
= g.GetHdc();
InBlock.gif         myDoc.body.setAttribute(
"scrollTop", (screenHeight - 5* i, 0);
InBlock.gif         brwTop 
= (int)myDoc.body.getAttribute("scrollTop"0);
InBlock.gif         PrintWindow(hwnd, hdc, 
0);
InBlock.gif         g.ReleaseHdc(hdc);
InBlock.gif         g.Flush();
InBlock.gif         screenfrag 
= Image.FromHbitmap(bm.GetHbitmap());
InBlock.gif         g2.DrawImage(screenfrag, brwLeft 
+ URLExtraLeft, brwTop + 
InBlock.gif            URLExtraHeight);
ExpandedSubBlockEnd.gif     }

InBlock.gif     
++myPageWidth;
ExpandedBlockEnd.gif }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值