ATL COM组件,将指定DC句柄保存为BMP

该博客介绍了将指定DC保存为BMP位图的方法,适用于暴露自身表面的动态程序情况捕捉。详细给出了相关接口和类的声明与实现,包括属性设置、拷贝位图、获取位图信息、写BMP文件等功能,还包含错误处理代码。

用于将指定DC保存为BMP位图,适用于暴露自身表面的动态程序情况捕捉。

// hdctobmpX.h : ChdctobmpX 的声明

#pragma once
#include "resource.h"       // 主符号


// IhdctobmpX
[
 object,
 uuid("7C3141C9-A96E-4775-93C7-FEE42C738B94"),
 dual, helpstring("IhdctobmpX 接口"),
 pointer_default(unique)
]
__interface IhdctobmpX : IDispatch
{
 [propput, id(1), helpstring("属性 HWND")] HRESULT HWND([in] ULONG newVal);
 [propput, id(2), helpstring("属性 savepath")] HRESULT savepath([in] BSTR newVal);
 [id(3), helpstring("方法savetobmp")] HRESULT savetobmp(void);
 [propput, id(4), helpstring("属性 bwidth")] HRESULT bwidth([in] DOUBLE newVal);
 [propput, id(5), helpstring("属性 bheight")] HRESULT bheight([in] DOUBLE newVal);
 [id(6), helpstring("方法savetobmp2")] HRESULT savetobmp2(void);
};

 

// ChdctobmpX

[
 coclass,
 threading("apartment"),
 vi_progid("HDCtoBMP.hdctobmpX"),
 progid("HDCtoBMP.hdctobmpX.1"),
 version(1.0),
 uuid("F4A2F19D-AF6B-434A-A104-C4AA013E7B37"),
 helpstring("hdctobmpX Class")
]

 

class ATL_NO_VTABLE ChdctobmpX :
 public IhdctobmpX
{


protected:

 //保存路径
 _bstr_t  m_path;
   
 //传入句柄
    HWND m_hwnd;

 
  //位图宽
DOUBLE     m_Width;
 
  //位图高
 DOUBLE m_Height;

public:
    //构造
 ChdctobmpX(){}


 DECLARE_PROTECT_FINAL_CONSTRUCT()
 HRESULT FinalConstruct(){return S_OK;}
 
 //系构
 void FinalRelease(){}

public:

 //属性设置
 STDMETHOD(put_HWND)(ULONG newVal);
 STDMETHOD(put_savepath)(BSTR newVal);

 //拷贝位图
 STDMETHOD(savetobmp)(void);

 STDMETHOD(put_bwidth)(DOUBLE newVal);
 STDMETHOD(put_bheight)(DOUBLE newVal);


private:
 PBITMAPINFO ChdctobmpX::CreateBitmapInfoStruct(HWND hwnd, HBITMAP hBmp);
void CreateBMPFile(HWND hwnd, LPTSTR pszFile, PBITMAPINFO pbi,HBITMAP hBMP, HDC hDC);
void Error( TCHAR * pText );

 

 

 


public:
 STDMETHOD(savetobmp2)(void);
};

---------------------------------------------------------------------------------------------------------

// hdctobmpX.cpp : ChdctobmpX 的实现

#include "stdafx.h"
#include "hdctobmpX.h"


// ChdctobmpX

 

//存BMP
STDMETHODIMP ChdctobmpX::savetobmp(void)
{
     // 设置图象保存位置加上.BMP扩展名
      
        TCHAR m_ShortName[MAX_PATH];

        wsprintf( m_ShortName, TEXT("%s.bmp/0"),(LPCTSTR)m_path);

  
  
      //获取输入当前DC
      HDC hdc=GetDC(this->m_hwnd);
            //由当前DC转化创建一完备DC
        HDC hdccmp = CreateCompatibleDC(hdc);
 
         //由当前DC创建位图DC
        HBITMAP hbmp = CreateCompatibleBitmap(hdc,m_Width, m_Height);
   if (hbmp == 0)  {return 0;}

         //完备DC选择位图DC
      if(!SelectObject(hdccmp, hbmp)){return 0;}

   //当前DC拷贝到完备DC
   BitBlt(hdccmp,0,0,m_Width, m_Height,hdc,0,0,SRCCOPY);


   PBITMAPINFO pbi=CreateBitmapInfoStruct(this->m_hwnd ,hbmp);

         CreateBMPFile(this->m_hwnd, m_ShortName,pbi,hbmp,hdccmp) ;

 

 return S_OK;
}


STDMETHODIMP ChdctobmpX::savetobmp2(void)
{
  // 设置图象保存位置加上.BMP扩展名
      
        TCHAR m_ShortName[MAX_PATH];

        wsprintf( m_ShortName, TEXT("%s.bmp/0"),(LPCTSTR)m_path);

  
  
      //获取输入当前DC
  HDC hdc=GetWindowDC(this->m_hwnd);
            //由当前DC转化创建一完备DC
        HDC hdccmp = CreateCompatibleDC(hdc);
 
         //由当前DC创建位图DC
        HBITMAP hbmp = CreateCompatibleBitmap(hdc,m_Width, m_Height);
   if (hbmp == 0)  {return 0;}

         //完备DC选择位图DC
      if(!SelectObject(hdccmp, hbmp)){return 0;}

   //当前DC拷贝到完备DC
   BitBlt(hdccmp,0,0,m_Width, m_Height,hdc,0,0,SRCCOPY);


   PBITMAPINFO pbi=CreateBitmapInfoStruct(this->m_hwnd ,hbmp);

   CreateBMPFile(this->m_hwnd, m_ShortName,pbi,hbmp,hdccmp) ;

 

 return S_OK;

}

 


/////////////////////////////属性///////////////////////////
STDMETHODIMP ChdctobmpX::put_bwidth(DOUBLE newVal)
{
 // TODO: 在此添加实现代码
 this->m_Width=newVal;
 return S_OK;
}

STDMETHODIMP ChdctobmpX::put_bheight(DOUBLE newVal)
{
 // TODO: 在此添加实现代码
 this->m_Height=newVal;
 return S_OK;
}

 

 


STDMETHODIMP ChdctobmpX::put_savepath(BSTR newVal)
{
 // TODO: 在此添加实现代码
    this->m_path= newVal;
 SysFreeString(newVal);
 return S_OK;
}
STDMETHODIMP ChdctobmpX::put_HWND(ULONG newVal)
{
 this->m_hwnd=(HWND)newVal;
 return S_OK;
}

 

 

 


////////////////////////私有成员////////////////////////
//获取位图信息
PBITMAPINFO ChdctobmpX::CreateBitmapInfoStruct(HWND hwnd, HBITMAP hBmp)
{
    BITMAP bmp;
    PBITMAPINFO pbmi;
    WORD    cClrBits;

    // Retrieve the bitmap color format, width, and height.
    if (!GetObject(hBmp, sizeof(BITMAP), (LPSTR)&bmp))
        Error("GetObject");

    // Convert the color format to a count of bits.
    cClrBits = (WORD)(bmp.bmPlanes * bmp.bmBitsPixel);
    if (cClrBits == 1)
        cClrBits = 1;
    else if (cClrBits <= 4)
        cClrBits = 4;
    else if (cClrBits <= 8)
        cClrBits = 8;
    else if (cClrBits <= 16)
        cClrBits = 16;
    else if (cClrBits <= 24)
        cClrBits = 24;
    else cClrBits = 32;

    // Allocate memory for the BITMAPINFO structure. (This structure
    // contains a BITMAPINFOHEADER structure and an array of RGBQUAD
    // data structures.)

     if (cClrBits != 24)
         pbmi = (PBITMAPINFO) LocalAlloc(LPTR,
                    sizeof(BITMAPINFOHEADER) +
                    sizeof(RGBQUAD) * (1<< cClrBits));

     // There is no RGBQUAD array for the 24-bit-per-pixel format.

     else
         pbmi = (PBITMAPINFO) LocalAlloc(LPTR,
                    sizeof(BITMAPINFOHEADER));

    // Initialize the fields in the BITMAPINFO structure.

    pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
    pbmi->bmiHeader.biWidth = bmp.bmWidth;
    pbmi->bmiHeader.biHeight = bmp.bmHeight;
    pbmi->bmiHeader.biPlanes = bmp.bmPlanes;
    pbmi->bmiHeader.biBitCount = bmp.bmBitsPixel;
    if (cClrBits < 24)
        pbmi->bmiHeader.biClrUsed = (1<<cClrBits);

    // If the bitmap is not compressed, set the BI_RGB flag.
    pbmi->bmiHeader.biCompression = BI_RGB;

    // Compute the number of bytes in the array of color
    // indices and store the result in biSizeImage.
    // For Windows NT, the width must be DWORD aligned unless
    // the bitmap is RLE compressed. This example shows this.
    // For Windows 95/98/Me, the width must be WORD aligned unless the
    // bitmap is RLE compressed.
    pbmi->bmiHeader.biSizeImage = ((pbmi->bmiHeader.biWidth * cClrBits +31) & ~31) /8
                                  * pbmi->bmiHeader.biHeight;
    // Set biClrImportant to 0, indicating that all of the
    // device colors are important.
     pbmi->bmiHeader.biClrImportant = 0;
     return pbmi;
 }

 

//写BMP文件
void ChdctobmpX::CreateBMPFile(HWND hwnd, LPTSTR pszFile, PBITMAPINFO pbi,HBITMAP hBMP, HDC hDC)
 {
     HANDLE hf;                 // file handle
    BITMAPFILEHEADER hdr;       // bitmap file-header
    PBITMAPINFOHEADER pbih;     // bitmap info-header
    LPBYTE lpBits;              // memory pointer
    DWORD dwTotal;              // total count of bytes
    DWORD cb;                   // incremental count of bytes
    BYTE *hp;                   // byte pointer
    DWORD dwTmp;

    pbih = (PBITMAPINFOHEADER) pbi;
    lpBits = (LPBYTE) GlobalAlloc(GMEM_FIXED, pbih->biSizeImage);

    if (!lpBits)
         Error("GlobalAlloc");

    // Retrieve the color table (RGBQUAD array) and the bits
    // (array of palette indices) from the DIB.
    if (!GetDIBits(hDC, hBMP, 0, (WORD) pbih->biHeight, lpBits, pbi,
        DIB_RGB_COLORS))
    {
        Error("GetDIBits");
    }

    // Create the .BMP file.
    hf = CreateFile(pszFile,
                   GENERIC_READ | GENERIC_WRITE,
                   (DWORD) 0,
                    NULL,
                   CREATE_ALWAYS,
                   FILE_ATTRIBUTE_NORMAL,
                   (HANDLE) NULL);
    if (hf == INVALID_HANDLE_VALUE)
        Error("CreateFile");
    hdr.bfType = 0x4d42;        // 0x42 = "B" 0x4d = "M"
    // Compute the size of the entire file.
    hdr.bfSize = (DWORD) (sizeof(BITMAPFILEHEADER) +
                 pbih->biSize + pbih->biClrUsed
                 * sizeof(RGBQUAD) + pbih->biSizeImage);
    hdr.bfReserved1 = 0;
    hdr.bfReserved2 = 0;

    // Compute the offset to the array of color indices.
    hdr.bfOffBits = (DWORD) sizeof(BITMAPFILEHEADER) +
                    pbih->biSize + pbih->biClrUsed
                    * sizeof (RGBQUAD);

    // Copy the BITMAPFILEHEADER into the .BMP file.
    if (!WriteFile(hf, (LPVOID) &hdr, sizeof(BITMAPFILEHEADER),
        (LPDWORD) &dwTmp,  NULL))
    {
       Error("WriteFile");
    }

    // Copy the BITMAPINFOHEADER and RGBQUAD array into the file.
    if (!WriteFile(hf, (LPVOID) pbih, sizeof(BITMAPINFOHEADER)
                  + pbih->biClrUsed * sizeof (RGBQUAD),
      (LPDWORD) &dwTmp, ( NULL))) {
       Error("WriteFile"); }

    // Copy the array of color indices into the .BMP file.
    dwTotal = cb = pbih->biSizeImage;
    hp = lpBits;
    if (!WriteFile(hf, (LPSTR) hp, (int) cb, (LPDWORD) &dwTmp,NULL))
           Error("WriteFile");

    // Close the .BMP file.
     if (!CloseHandle(hf))
           Error("CloseHandle");

    // Free memory.
    GlobalFree((HGLOBAL)lpBits);
}

 

 //错误显示
 void ChdctobmpX::Error( TCHAR * pText )
{
  
    ::MessageBox( NULL, pText, TEXT("Error!"), MB_OK | MB_TASKMODAL | MB_SETFOREGROUND );
}

// MFCPaintView.cpp: CMFCPaintView 类的实现 // #include "pch.h" #include "framework.h" // SHARED_HANDLERS 可以在实现预览、缩略图和搜索筛选器句柄的 // ATL 项目中进行定义,并允许与该项目共享文档代码。 #ifndef SHARED_HANDLERS #include "MFCPaint.h" #endif #include "MFCPaintDoc.h" #include "MFCPaintView.h" #include "CMy_Global.h" #ifdef _DEBUG #define new DEBUG_NEW #endif // CMFCPaintView IMPLEMENT_DYNCREATE(CMFCPaintView, CView) BEGIN_MESSAGE_MAP(CMFCPaintView, CView) // 标准打印命令 ON_COMMAND(ID_FILE_PRINT, &CView::OnFilePrint) ON_COMMAND(ID_FILE_PRINT_DIRECT, &CView::OnFilePrint) ON_COMMAND(ID_FILE_PRINT_PREVIEW, &CView::OnFilePrintPreview) ON_WM_LBUTTONDOWN() ON_WM_LBUTTONUP() ON_WM_MOUSEMOVE() END_MESSAGE_MAP() // CMFCPaintView 构造/析构 CMFCPaintView::CMFCPaintView() noexcept { // TODO: 在此处添加构造代码 isPainting = false; isLoading = false; isWriting = false; } CMFCPaintView::~CMFCPaintView() { } BOOL CMFCPaintView::PreCreateWindow(CREATESTRUCT& cs) { // TODO: 在此处通过修改 // CREATESTRUCT cs 来修改窗口类或样式 return CView::PreCreateWindow(cs); } // CMFCPaintView 绘图 void CMFCPaintView::OnDraw(CDC* /*pDC*/) { CMFCPaintDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); if (!pDoc) return; // TODO: 在此处为本机数据添加绘制代码 } // CMFCPaintView 打印 BOOL CMFCPaintView::OnPreparePrinting(CPrintInfo* pInfo) { // 默认准备 return DoPreparePrinting(pInfo); } void CMFCPaintView::OnBeginPrinting(CDC* /*pDC*/, CPrintInfo* /*pInfo*/) { // TODO: 添加额外的打印前进行的初始化过程 } void CMFCPaintView::OnEndPrinting(CDC* /*pDC*/, CPrintInfo* /*pInfo*/) { // TODO: 添加打印后进行的清理过程 } // CMFCPaintView 诊断 #ifdef _DEBUG void CMFCPaintView::AssertValid() const { CView::AssertValid(); } void CMFCPaintView::Dump(CDumpContext& dc) const { CView::Dump(dc); } CMFCPaintDoc* CMFCPaintView::GetDocument() const // 非调试版本是内联的 { ASSERT(m_pDocument->IsKindOf(RUNTIME_CLASS(CMFCPaintDoc))); return (CMFCPaintDoc*)m_pDocument; } #endif //_DEBUG // CMFCPaintView 消息处理程序 void CMFCPaintView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: 在此添加消息处理程序代码和/或调用默认值 P_Start = point; isPainting = true; CView::OnLButtonDown(nFlags, point); } void CMFCPaintView::OnLButtonUp(UINT nFlags, CPoint point) { // TODO: 在此添加消息处理程序代码和/或调用默认值 if (isWriting) { CDC* pdc = GetDC(); CFont new_font; //创建字体宋体格式 100为字高 VERIFY(new_font.CreatePointFont(250, _T("宋体"), pdc)); //选择该字体进入PDC CFont* default_font = pdc->SelectObject(&new_font); //设置字体颜色 pdc->SetTextColor(Line_Color); pdc->TextOut(point.x, point.y, Text); isWriting = false; isPainting = false; return; } if (isLoading) { isLoading = false; return; } if (Shape_Type == 6) { isPainting = false; return; } P_End = point; CPen* pNewPen; CPen* pOldPen; CBrush* pNewBrush1; CBrush* pOldBrush1; CDC* pdc = GetDC(); if (Shape_Type == 4) { pNewPen = new CPen; pNewPen->CreatePen(Line_Type, Line_Width, Line_Color); pOldPen = pdc->SelectObject(pNewPen); //选择新笔,并保存旧笔 //画矩形 pdc->Ellipse(P_Start.x, P_Start.y, P_End.x, P_End.y); //恢复原有的笔 pdc->SelectObject(pOldPen); delete pNewPen; if (isfill) { pNewBrush1 = new CBrush; pNewBrush1->CreateSolidBrush(Fill_Color); //选择新画刷 pOldBrush1 = pdc->SelectObject(pNewBrush1); //绘制圆 pdc->Ellipse(P_Start.x, P_Start.y, P_End.x, P_End.y); //恢复原有画刷 pdc->SelectObject(pOldBrush1); delete pNewBrush1; } } else if (Shape_Type == 2) { pNewPen = new CPen; pNewPen->CreatePen(Line_Type, Line_Width, Line_Color); pOldPen = pdc->SelectObject(pNewPen); //选择新笔,并保存旧笔 //画矩形 pdc->Rectangle(P_Start.x, P_Start.y, P_End.x, P_End.y); //恢复原有的笔 pdc->SelectObject(pOldPen); delete pNewPen; if (isfill) { pNewBrush1 = new CBrush; pNewBrush1->CreateSolidBrush(Fill_Color); //选择新画刷 pOldBrush1 = pdc->SelectObject(pNewBrush1); //绘制矩形 pdc->Rectangle(P_Start.x, P_Start.y, P_End.x, P_End.y); //恢复原有画刷 pdc->SelectObject(pOldBrush1); delete pNewBrush1; } } else if (Shape_Type == 1) { pNewPen = new CPen; pNewPen->CreatePen(Line_Type, Line_Width, Line_Color); pOldPen = pdc->SelectObject(pNewPen); //选择新笔,并保存旧笔 //画直线 pdc->MoveTo(P_Start.x, P_Start.y); pdc->LineTo(P_End.x, P_End.y); //恢复原有的笔 pdc->SelectObject(pOldPen); delete pNewPen; } else if (Shape_Type == 3) { pNewPen = new CPen; pNewPen->CreatePen(Line_Type, Line_Width, Line_Color); pOldPen = pdc->SelectObject(pNewPen); //选择新笔,并保存旧笔 //画矩形 int d = P_End.x - P_Start.x; pdc->Ellipse(P_Start.x, P_Start.y, P_Start.x + d, P_Start.y + d); //恢复原有的笔 pdc->SelectObject(pOldPen); delete pNewPen; if (isfill) { pNewBrush1 = new CBrush; pNewBrush1->CreateSolidBrush(Fill_Color); //选择新画刷 pOldBrush1 = pdc->SelectObject(pNewBrush1); //绘制圆 pdc->Ellipse(P_Start.x, P_Start.y, P_Start.x + d, P_Start.y + d); //恢复原有画刷 pdc->SelectObject(pOldBrush1); delete pNewBrush1; } } else if (Shape_Type == 5) { pNewPen = new CPen; pNewPen->CreatePen(Line_Type, Line_Width, Line_Color); pOldPen = pdc->SelectObject(pNewPen); //选择新笔,并保存旧笔 // 定义一个矩形,用于确定弧线的边界 CRect rect(P_Start.x, P_Start.y, P_End.x, P_End.y); // 绘制一条从point1到point2的弧线 pdc->Arc(rect, P_Start, P_End); //恢复原有的笔 pdc->SelectObject(pOldPen); delete pNewPen; } isPainting = false; CView::OnLButtonUp(nFlags, point); } void CMFCPaintView::OnMouseMove(UINT nFlags, CPoint point) { // TODO: 在此添加消息处理程序代码和/或调用默认值 if (Shape_Type == 6 && isPainting) { CPen* pNewPen; CDC* pdc = GetDC(); pNewPen = new CPen; pNewPen->CreatePen(0, Line_Width, Line_Color); pdc->SelectObject(pNewPen); //选择新笔,并保存旧笔 pdc->MoveTo(P_Start); pdc->LineTo(point); P_Start = point; delete pNewPen; } CView::OnMouseMove(nFlags, point); } void CMFCPaintView::ClearDrawing() { // 示例代码:将绘图缓冲区清空为白色背景 CClientDC dc(this); CRect rect; GetClientRect(&rect); dc.FillSolidRect(rect, RGB(255, 255, 255)); } void CMFCPaintView::LoadMyImage() { CFileDialog fileDlg(TRUE, _T("png"), NULL, 0, _T("image Files(*.bmp; *.jpeg;*.jpg;*.png)|*.JPG;*.PNG;*.BMP|All Files (*.*) |*.*|"), this); fileDlg.DoModal(); CString StrFilePath = fileDlg.GetPathName(); if (StrFilePath == _T("")) { return; } CDC* pdc = GetDC(); CRect rect; GetClientRect(&rect); CImage image; image.Load(StrFilePath); image.BitBlt(*pdc, 40, 40, rect.Width(), rect.Height(), 0, 0); image.Destroy(); //没有Destroy()会有内存泄漏。Detach();不行的。 isLoading = true; } void CMFCPaintView::SavePainting() { CClientDC dc(this); CRect rect; //BOOL showMsgTag; //是否要弹出”图像保存成功对话框" GetClientRect(&rect); //获取画布大小 HBITMAP hbitmap = CreateCompatibleBitmap(dc, rect.right - rect.left, rect.bottom - rect.top); //创建兼容位图 HDC hdc = CreateCompatibleDC(dc); //创建兼容DC,以便将图像保存为不同的格式 HBITMAP hOldMap = (HBITMAP)SelectObject(hdc, hbitmap); //将位图选入DC,并保存返回值 BitBlt(hdc, 0, 0, rect.right - rect.left, rect.bottom - rect.top, dc, 0, 0, SRCCOPY); //将屏幕DC的图像复制到内存DC中 CImage image; image.Attach(hbitmap); //将位图化为一般图像 CString strFilter = _T("位图文件(*.bmp)|*.bmp|JPEG 图像文件|*.jpg|GIF图像文件 | *.gif | PNG图像文件 | *.png |其他格式(*.*) | *.* || "); CFileDialog dlg(FALSE, _T("bmp"), _T("iPaint1.bmp"), NULL, strFilter); if (dlg.DoModal() != IDOK) return; CString strFileName; //如果用户没有指定文件扩展名,则为其添加一个 CString strExtension; strFileName = dlg.m_ofn.lpstrFile; if (dlg.m_ofn.nFileExtension == 0) //扩展名项目为0 { switch (dlg.m_ofn.nFilterIndex) { case 1: strExtension = "bmp"; break; case 2: strExtension = "jpg"; break; case 3: strExtension = "gif"; break; case 4: strExtension = "png"; break; default: break; } strFileName = strFileName + "." + strExtension; } saveFilePath = strFileName; //saveFilePath为视类中的全局变量,类型为CString HRESULT hResult = image.Save(saveFilePath); //保存图像 if (FAILED(hResult)) { MessageBox(_T("保存图像文件失败!")); } else { MessageBox(_T("文件保存成功!")); } image.Detach(); SelectObject(hdc, hOldMap); } 帮我改成项目名为202402172226lgqsx
06-20
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值