VC中实现截屏并保存为.bmp格式的图片


HBITMAP CHtfFundSalesSysDlg::CopyScreenToBitmap(LPRECT lpRect) //lpRect 代表选定区域

{
HDC hScrDC, hMemDC;   
// 屏幕和内存设备描述表
HBITMAP hBitmap,hOldBitmap;   
// 位图句柄
int nX, nY, nX2, nY2;   
// 选定区域坐标
int nWidth, nHeight;   
// 位图宽度和高度
int xScrn, yScrn;   
// 屏幕分辨率
// 确保选定区域不为空矩形
if (IsRectEmpty(lpRect))
return NULL;
//为屏幕创建设备描述表
hScrDC = CreateDC(_T("DISPLAY"), NULL, NULL, NULL);
//为屏幕设备描述表创建兼容的内存设备描述表
hMemDC = CreateCompatibleDC(hScrDC);
// 获得选定区域坐标
nX = lpRect->left;
nY = lpRect->top;
nX2 = lpRect->right;
nY2 = lpRect->bottom;
// 获得屏幕分辨率
xScrn = GetDeviceCaps(hScrDC, HORZRES);
yScrn = GetDeviceCaps(hScrDC, VERTRES);
//确保选定区域是可见的
if (nX < 0)
nX = 0;
if (nY < 0)
nY = 0;
if (nX2 > xScrn)
nX2 = xScrn;
if (nY2 > yScrn)
nY2 = yScrn;
nWidth = nX2 - nX;
nHeight = nY2 - nY;
// 创建一个与屏幕设备描述表兼容的位图
hBitmap=CreateCompatibleBitmap(hScrDC,nWidth,nHeight);
// 把新位图选到内存设备描述表中
hOldBitmap=(HBITMAP)SelectObject(hMemDC,hBitmap);
// 把屏幕设备描述表拷贝到内存设备描述表中
BitBlt(hMemDC,0,0, nWidth,nHeight,hScrDC, nX, nY, SRCCOPY);
//得到屏幕位图的句柄
hBitmap=(HBITMAP)SelectObject(hMemDC,hOldBitmap);
//清除
DeleteDC(hScrDC);
DeleteDC(hMemDC);
// 返回位图句柄
return hBitmap;
}


int CHtfFundSalesSysDlg::SaveBitmapToFile(HBITMAP hBitmap, LPWSTR lpFileName) //hBitmap 为刚才的屏幕位图句柄
{ //lpFileName 为位图文件名
 HDC hDC;   
  //设备描述表
 int iBits;   
 //当前显示分辨率下每个像素所占字节数
 WORD wBitCount;   
  //位图中每个像素所占字节数
 //定义调色板大小, 位图中像素字节大小 , 位图文件大小 , 写入文件字节数
 DWORD dwPaletteSize=0,dwBmBitsSize,dwDIBSize, dwWritten;
 BITMAP Bitmap;   
 //位图属性结构
 BITMAPFILEHEADER bmfHdr;   
 //位图文件头结构
 BITMAPINFOHEADER bi;   
 //位图信息头结构
 LPBITMAPINFOHEADER lpbi;   
 //指向位图信息头结构
  HANDLE fh, hDib, hPal;
 HPALETTE hOldPal=NULL;
 //定义文件,分配内存句柄,调色板句柄
  
 //计算位图文件每个像素所占字节数
 hDC = CreateDC(_T("DISPLAY"),NULL,NULL,NULL);
 iBits = GetDeviceCaps(hDC, BITSPIXEL) *
  GetDeviceCaps(hDC, PLANES);
 DeleteDC(hDC);
 if (iBits <= 1)
  wBitCount = 1;
 else if (iBits <= 4)
  wBitCount = 4;
 else if (iBits <= 8)
  wBitCount = 8;
 else if (iBits <= 24)
  wBitCount = 24;
else
  wBitCount = 32;
 //计算调色板大小
 if (wBitCount <= 8)
  dwPaletteSize=(1<<wBitCount)*sizeof(RGBQUAD);
  
 //设置位图信息头结构
 GetObject(hBitmap, sizeof(BITMAP), (LPSTR)&Bitmap);
 bi.biSize = sizeof(BITMAPINFOHEADER);
 bi.biWidth = Bitmap.bmWidth;
 bi.biHeight = Bitmap.bmHeight;
 bi.biPlanes = 1;
 bi.biBitCount = wBitCount;
 bi.biCompression = BI_RGB;
 bi.biSizeImage = 0;
 bi.biXPelsPerMeter = 0;
 bi.biYPelsPerMeter = 0;
 bi.biClrUsed = 0;
 bi.biClrImportant = 0;
  
 dwBmBitsSize = ((Bitmap.bmWidth*wBitCount+31)/32)*4*Bitmap.bmHeight;
 //为位图内容分配内存


/*xxxxxxxx计算位图大小分解一下(解释一下上面的语句)xxxxxxxxxxxxxxxxxxxx  
//每个扫描行所占的字节数应该为4的整数倍,具体算法为:
 int biWidth = (Bitmap.bmWidth*wBitCount) / 32;
 if((Bitmap.bmWidth*wBitCount) % 32)
  biWidth++; //不是整数倍的加1
 biWidth *= 4;//到这里,计算得到的为每个扫描行的字节数。
 dwBmBitsSize = biWidth * Bitmap.bmHeight;//得到大小
xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx*/




 hDib = GlobalAlloc(GHND,dwBmBitsSize+dwPaletteSize+sizeof(BITMAPINFOHEADER));
 lpbi = (LPBITMAPINFOHEADER)GlobalLock(hDib);
 *lpbi = bi;
 // 处理调色板   
 hPal = GetStockObject(DEFAULT_PALETTE);
 if (hPal)
 {
  hDC = ::GetDC(NULL);
  hOldPal=SelectPalette(hDC,(HPALETTE)hPal,FALSE);
  RealizePalette(hDC);
 }
 // 获取该调色板下新的像素值
 GetDIBits(hDC,hBitmap,0,(UINT)Bitmap.bmHeight,(LPSTR)lpbi+sizeof(BITMAPINFOHEADER)+dwPaletteSize, (BITMAPINFO *)lpbi,DIB_RGB_COLORS);
 //恢复调色板   
 if (hOldPal)
 {
  SelectPalette(hDC, hOldPal, TRUE);
  RealizePalette(hDC);
  ::ReleaseDC(NULL, hDC);
 }
 //创建位图文件   
 fh=CreateFile(lpFileName, GENERIC_WRITE,0, NULL, CREATE_ALWAYS,FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN, NULL);
 if (fh==INVALID_HANDLE_VALUE)
  return FALSE;
 // 设置位图文件头
 bmfHdr.bfType = 0x4D42; // "BM"
 dwDIBSize=sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER)+dwPaletteSize+dwBmBitsSize;  
 bmfHdr.bfSize = dwDIBSize;
 bmfHdr.bfReserved1 = 0;
 bmfHdr.bfReserved2 = 0;
 bmfHdr.bfOffBits = (DWORD)sizeof(BITMAPFILEHEADER)+(DWORD)sizeof(BITMAPINFOHEADER)+dwPaletteSize;
 // 写入位图文件头
 WriteFile(fh, (LPSTR)&bmfHdr, sizeof(BITMAPFILEHEADER), &dwWritten, NULL);
 // 写入位图文件其余内容
  WriteFile(fh, (LPSTR)lpbi, sizeof(BITMAPINFOHEADER)+dwPaletteSize+dwBmBitsSize , &dwWritten, NULL);
 //清除   
 GlobalUnlock(hDib);
 GlobalFree(hDib);
 CloseHandle(fh);
 return TRUE;

}


应用例子:

CFileDialog fileDialog( FALSE,_T("*.bmp"),NULL,NULL,_T("图片文件(*.bmp)|*.bmp||"));
if (fileDialog.DoModal() == IDOK)
{
m_strSavePath = fileDialog.GetPathName();
}


CRect rect;

GetWindowRect(&rect);

HBitmap bitmap = CopyScreenToBitmap(rect);

SaveBitmapToFile(bitmap, m_strSavePath);

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
VC 实现 HTTP 的 POST 方式上传文件并获取返回值,可以通过以下步骤实现: 1. 引用 WinINet 库 在 VC 项目,需要引用 WinINet 库,可以通过以下步骤添加: 在 Solution Explorer ,右键单击项目,选择 Properties。 在 Property Pages 对话框,选择 Configuration Properties -> Linker -> Input。 在 Additional Dependencies 添加 WinINet.lib。 2. 创建 HTTP 连接 使用 InternetOpen 函数创建 HTTP 连接,该函数返回一个 HINTERNET 句柄,用于后续操作: ``` HINTERNET hInternet = InternetOpen(_T("MyApp"), INTERNET_OPEN_TYPE_PRECONFIG, NULL, NULL, 0); ``` 其,第一个参数是应用程序名称,第二个参数指定连接类型,第三个和第四个参数分别为代理服务器的名称和端口号,最后一个参数为可选的标志,如果为 0,则使用默认标志。 3. 创建 HTTP 请求 使用 HttpOpenRequest 函数创建 HTTP 请求,该函数需要传入上一步创建的 HINTERNET 句柄,返回一个 HINTERNET 句柄,用于后续操作: ``` HINTERNET hHttp = HttpOpenRequest(hInternet, _T("POST"), _T("/upload.php"), NULL, NULL, NULL, INTERNET_FLAG_RELOAD, 0); ``` 其,第一个参数为上一步创建的 HINTERNET 句柄,第二个参数指定请求方法,这里使用 POST 方法,第三个参数为上传文件的路径,第四个和第五个参数分别为 HTTP 请求头和请求头的长度,最后一个参数为可选的标志,这里使用 INTERNET_FLAG_RELOAD 标志,表示每次请求都从服务器获取最新的内容。 4. 添加 HTTP 请求头 使用 HttpAddRequestHeaders 函数添加 HTTP 请求头,这里需要添加 Content-Type 和 Content-Length 请求头: ``` TCHAR szContentType[] = _T("Content-Type: multipart/form-data; boundary=***"); HttpAddRequestHeaders(hHttp, szContentType, _tcslen(szContentType), HTTP_ADDREQ_FLAG_ADD); TCHAR szContentLength[256]; sprintf_s(szContentLength, _T("Content-Length: %d"), dwTotalSize); HttpAddRequestHeaders(hHttp, szContentLength, strlen(szContentLength), HTTP_ADDREQ_FLAG_ADD); ``` 其,第一个参数为上一步创建的 HINTERNET 句柄,第二个参数为请求头的内容,第三个参数为请求头的长度,最后一个参数为可选的标志,这里使用 HTTP_ADDREQ_FLAG_ADD 标志,表示添加请求头而不是替换。 5. 构建 HTTP 请求正文 使用 HttpSendRequest 函数发送 HTTP 请求,这里需要构建 HTTP 请求正文,即上传文件的数据,可以使用 multipart/form-data 格式,该格式需要指定一个分隔符,每个字段都以分隔符开始,以分隔符结束,字段名和字段值之间用两个换行符分隔,文件数据需要以二进制方式传输,每个文件以分隔符开始,以分隔符结束,文件名和文件数据之间用两个换行符分隔。 ``` TCHAR szBoundary[] = _T("***"); TCHAR szHeader[1024]; sprintf_s(szHeader, _T("--%s\r\nContent-Disposition: form-data; name=\"file\"; filename=\"%s\"\r\nContent-Type: application/octet-stream\r\n\r\n"), szBoundary, szFileName); DWORD dwHeaderSize = strlen(szHeader); TCHAR szFooter[1024]; sprintf_s(szFooter, _T("\r\n--%s--\r\n"), szBoundary); DWORD dwFooterSize = strlen(szFooter); DWORD dwTotalSize = dwHeaderSize + dwFileSize + dwFooterSize; HGLOBAL hGlobal = GlobalAlloc(GMEM_MOVEABLE, dwTotalSize); LPBYTE pBuffer = (LPBYTE)GlobalLock(hGlobal); memcpy(pBuffer, szHeader, dwHeaderSize); pBuffer += dwHeaderSize; DWORD dwBytesRead; ReadFile(hFile, pBuffer, dwFileSize, &dwBytesRead, NULL); pBuffer += dwBytesRead; memcpy(pBuffer, szFooter, dwFooterSize); pBuffer += dwFooterSize; GlobalUnlock(hGlobal); HttpSendRequest(hHttp, szContentType, _tcslen(szContentType), hGlobal, dwTotalSize); ``` 其,第一个参数为上一步创建的 HINTERNET 句柄,第二个参数为请求头的内容,第三个参数为请求头的长度,第四个参数为上传文件的数据,第五个参数为上传文件的数据长度。 6. 获取 HTTP 响应 使用 InternetReadFile 函数获取 HTTP 响应,这里需要读取 HTTP 响应正文,即服务器返回的数据: ``` BYTE szBuffer[1024]; DWORD dwBytesRead; while (InternetReadFile(hHttp, szBuffer, sizeof(szBuffer), &dwBytesRead) && dwBytesRead > 0) { // 处理读取到的数据 } ``` 其,第一个参数为上一步创建的 HINTERNET 句柄,第二个参数为读取数据的缓冲区,第三个参数为缓冲区的长度,第四个参数为读取到的数据长度。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值