欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++實(shí)現(xiàn)屏幕截圖

 更新時(shí)間:2018年05月15日 14:18:44   作者:sunflover454  
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)屏幕截圖功能,截圖自動(dòng)保存為png格式文件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

上回分享了一個(gè)全屏截圖的代碼,保存為BMP,參考:C++實(shí)現(xiàn)屏幕截圖(全屏截圖)

實(shí)際使用的過(guò)程中我發(fā)現(xiàn)截圖文件實(shí)在大,無(wú)奈又整成了PNG截圖,現(xiàn)在分享出來(lái)。

MakePNG.h

//MakePNG.h 
 
#pragma once 
#include <GdiPlus.h> 
using namespace Gdiplus; 
#pragma comment(lib,"GdiPlus.lib") 
 
class CMakePNG 
{ 
public: 
 CMakePNG(void); 
 ~CMakePNG(void); 
 
 BOOL MakePNG(HDC hDC,CRect rect,CString strFilePath); 
 BOOL BMptoPNG(LPCWSTR StrBMp,LPCWSTR StrPNG); 
 BOOL PNGtoBMp(LPCWSTR StrPNG,LPCWSTR StrBMp); 
 BOOL GetEncoderClsid(WCHAR* pFormat,CLSID* pClsid); 
private: 
 GdiplusStartupInput m_gdiplusStartupInput; 
 ULONG_PTR m_pGdiToken; 
}; 

MakePNG.cpp

//MakePNG.cpp 
 
#include "StdAfx.h" 
#include "MakePNG.h" 
 
CMakePNG::CMakePNG(void) 
{ 
 GdiplusStartup(&m_pGdiToken,&m_gdiplusStartupInput,NULL); 
} 
 
CMakePNG::~CMakePNG(void) 
{ 
} 
 
/***************************************************************************/ 
/* 功能: 根據(jù)rect屏幕抓圖,保存為文件名為strFilePath的PNG圖像文件 */ 
/* 輸入?yún)?shù): HDC hDC   屏幕HDC;    */ 
/*  CRect rect  需要的矩形;   */ 
/*  CString strFilePath 保存文件全路徑(含后綴名); */ 
/***************************************************************************/ 
BOOL CMakePNG::MakePNG(HDC hDC, CRect rect, CString strFilePath) 
{ 
 BITMAP bmp; 
 PBITMAPINFO pbmi; 
 PBITMAPINFOHEADER pbih; // bitmap info-header 
 BITMAPFILEHEADER hdr; // bitmap file-header 
 WORD cClrBits; 
 LPBYTE lpBits;  // memory pointer 
 DWORD dwTmp; 
 DWORD cb;   // incremental count of bytes 
 BYTE *hp;   // byte pointer 
 HANDLE hfile;  // file handle 
 CString szBMPFilename = strFilePath.Left(strFilePath.GetLength() - 3) + _T("bmp");//先保存成位圖 
 HDC hdcCompatible = CreateCompatibleDC(hDC); 
 HBITMAP hbmScreen = CreateCompatibleBitmap(hDC, rect.Width(), rect.Height()); 
 
 if (hbmScreen == NULL) 
 { 
 AfxMessageBox(_T("CreateCompatibleBitmap() error")); 
 return FALSE; 
 } 
 
 // Select the bitmaps into the compatible DC. 
 
 if (!SelectObject(hdcCompatible, hbmScreen)) 
 { 
 AfxMessageBox(_T("Compatible Bitmap Selection error")); 
 return FALSE; 
 } 
 
 //Copy color data for the entire display into a 
 //bitmap that is selected into a compatible DC. 
 
 if (!BitBlt(hdcCompatible, 
 0,0, 
 rect.Width(), rect.Height(), 
 hDC, 
 rect.left,rect.top, 
 SRCCOPY)) 
 { 
 AfxMessageBox(_T("Screen to Compat Blt Failed")); 
 return FALSE; 
 } 
 
 // Retrieve the bitmap's color format, width, and height. 
 if (!GetObject(hbmScreen, sizeof(BITMAP), (LPSTR)&bmp)) 
 { 
 AfxMessageBox(_T("GetObject()出錯(cuò)!")); 
 return FALSE; 
 } 
 // 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. 
 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; 
 
 pbih = (PBITMAPINFOHEADER) pbmi; 
 lpBits = (LPBYTE) GlobalAlloc(GMEM_FIXED, pbih->biSizeImage); 
 
 if (!lpBits) 
 { 
 AfxMessageBox(_T("內(nèi)存分配錯(cuò)誤!")); 
 return FALSE; 
 } 
 // Retrieve the color table (RGBQUAD array) and the bits 
 // (array of palette indices) from the DIB. 
 if (!GetDIBits(hDC, hbmScreen, 0, (WORD) pbih->biHeight, lpBits, pbmi, 
 DIB_RGB_COLORS)) 
 { 
 AfxMessageBox(_T("GetDIBits() error")); 
 return FALSE; 
 } 
 
 // Create the .BMP file. 
 hfile = CreateFile(szBMPFilename, 
 GENERIC_READ | GENERIC_WRITE, 
 (DWORD) 0, 
 NULL, 
 CREATE_ALWAYS, 
 FILE_ATTRIBUTE_NORMAL, 
 (HANDLE) NULL); 
 if (hfile == INVALID_HANDLE_VALUE) 
 { 
  AfxMessageBox(_T("創(chuàng)建文件失敗")); 
 return false; 
 } 
 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(hfile, (LPVOID) &hdr, sizeof(BITMAPFILEHEADER), 
 (LPDWORD) &dwTmp, NULL)) 
 { 
  AfxMessageBox(_T("寫(xiě)B(tài)MP文件頭失敗")); 
 return FALSE; 
 } 
 
 // Copy the BITMAPINFOHEADER and RGBQUAD array into the file. 
 if (!WriteFile(hfile, (LPVOID) pbih, sizeof(BITMAPINFOHEADER) 
 + pbih->biClrUsed * sizeof (RGBQUAD), 
 (LPDWORD) &dwTmp, ( NULL))) 
 { 
  AfxMessageBox(_T("寫(xiě)B(tài)MP文件頭失敗")); 
 return FALSE; 
 } 
 
 // Copy the array of color indices into the .BMP file. 
 cb = pbih->biSizeImage; 
 hp = lpBits; 
 if (!WriteFile(hfile, (LPSTR) hp, (int) cb, (LPDWORD) &dwTmp,NULL)) 
 { 
 AfxMessageBox(_T("寫(xiě)入BMP文件失敗")); 
 return FALSE; 
 } 
 
 // Close the .BMP file. 
 if (!CloseHandle(hfile)) 
 { 
  AfxMessageBox(_T("Can't close BMP file.")); 
 } 
 
 // Free memory. 
 GlobalFree((HGLOBAL)lpBits); 
 
 //轉(zhuǎn)換成PNG 
 if(!BMptoPNG(szBMPFilename,strFilePath)) 
 { 
 DeleteFile(szBMPFilename); 
 return FALSE; 
 } 
 DeleteObject(hbmScreen); 
 DeleteFile(szBMPFilename); 
 return TRUE; 
} 
// //轉(zhuǎn)換BMP文件為PNG文件  
BOOL CMakePNG::BMptoPNG(LPCWSTR StrBMp,LPCWSTR StrPNG) 
{ 
 CLSID encoderClsid; 
 Status stat; 
 Image* image = NULL; 
 image = Bitmap::FromFile(StrBMp,TRUE); 
 if (!GetEncoderClsid(L"image/png",&encoderClsid)) 
 { 
 return FALSE; 
 } 
 stat = image->Save(StrPNG,&encoderClsid,NULL); 
 if (stat != Ok) 
 { 
 return FALSE; 
 } 
 delete image; 
 return TRUE; 
} 
 
// 功能描述: 轉(zhuǎn)換PNG文件為BMP文件 
BOOL CMakePNG::PNGtoBMp(LPCWSTR StrPNG,LPCWSTR StrBMp) 
{ 
 CLSID encoderClsid; 
 Status stat; 
 Image* pImage; 
 pImage = Bitmap::FromFile(StrPNG,TRUE); 
 if (!GetEncoderClsid(L"image/bmp",&encoderClsid)) 
 { 
 return FALSE; 
 } 
 stat = pImage->Save(StrBMp,&encoderClsid,NULL); 
 if (stat != Ok) 
 { 
 return FALSE; 
 } 
 delete pImage; 
 return TRUE; 
} 
 
BOOL CMakePNG::GetEncoderClsid(WCHAR* pFormat,CLSID* pClsid) 
{ 
 UINT num = 0,size = 0; 
 ImageCodecInfo* pImageCodecInfo = NULL; 
 GetImageEncodersSize(&num,&size); 
 if (size == 0) 
 { 
 return FALSE; 
 } 
 pImageCodecInfo = (ImageCodecInfo*)(malloc(size)); 
 if (pImageCodecInfo == NULL) 
 { 
 return FALSE; 
 } 
 GetImageEncoders(num,size,pImageCodecInfo); 
 BOOL bfound = FALSE; 
 for (UINT i = 0;!bfound && i < num; i++) 
 { 
 if (_wcsicmp(pImageCodecInfo[i].MimeType,pFormat) == 0) 
 { 
  *pClsid = pImageCodecInfo[i].Clsid; 
  bfound = TRUE; 
 } 
 } 
 free(pImageCodecInfo); 
 return bfound; 
} 

以上兩個(gè)文件實(shí)際上是CMakePNG類(lèi),使用時(shí)需要把他們添加到項(xiàng)目中,調(diào)用方法如下:

wstring GetAppPathW() 
{ 
 wchar_t szExePath[MAX_PATH] = {0}; 
 GetModuleFileNameW(NULL, szExePath, MAX_PATH); 
 wchar_t *pstr = wcsrchr(szExePath, '\\'); 
 memset(pstr + 1, 0, 2); 
 wstring strAppPath(szExePath); 
 return strAppPath; 
} 
 
// 屏幕截圖 
CString CDemoDlg::ScreenShot(void) 
{ 
 CWnd *pDesktop = GetDesktopWindow(); 
 CDC *pDC = pDesktop->GetDC(); 
 CRect rect; 
 //獲取窗口的大小 
 pDesktop->GetClientRect(&rect); 
 
 //保存到的文件名 
 CString strFileName(GetAppPathW().c_str()); 
 strFileName += _T("ScreenShot\\"); 
 CreateDirectory((LPCTSTR)strFileName,NULL); 
 CTime t = CTime::GetCurrentTime(); 
 CString tt = t.Format("%Y%m%d_%H%M%S"); 
 strFileName += tt; 
 strFileName += _T(".PNG"); 
 //保存為PNG 
 CMakePNG MakePNG; 
 MakePNG.MakePNG(pDC->m_hDC,rect,strFileName); 
 ReleaseDC(pDC); 
 return strFileName; 
} 

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • C語(yǔ)言二進(jìn)制思想以及數(shù)據(jù)的存儲(chǔ)

    C語(yǔ)言二進(jìn)制思想以及數(shù)據(jù)的存儲(chǔ)

    本文主要介紹了C語(yǔ)言的二進(jìn)制思想以及數(shù)據(jù)的存儲(chǔ),這里對(duì)二進(jìn)制和數(shù)據(jù)存儲(chǔ)做了詳細(xì)的說(shuō)明,對(duì)開(kāi)始學(xué)習(xí)C語(yǔ)言的同學(xué)比較有參考價(jià)值
    2016-07-07
  • C語(yǔ)言中g(shù)etchar函數(shù)詳解看這一篇就夠了(函數(shù)功能、使用、返回值)

    C語(yǔ)言中g(shù)etchar函數(shù)詳解看這一篇就夠了(函數(shù)功能、使用、返回值)

    getchar讀取字符的函數(shù),今天通過(guò)本文給大家介紹C語(yǔ)言中g(shù)etchar函數(shù)簡(jiǎn)介用法示例詳解,感興趣的朋友跟隨小編一起看看吧
    2023-02-02
  • 從匯編看c++中的多態(tài)詳解

    從匯編看c++中的多態(tài)詳解

    下面小編就為大家?guī)?lái)一篇從匯編看c++中的多態(tài)詳解。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2016-06-06
  • C語(yǔ)言實(shí)現(xiàn)騎士飛行棋小游戲

    C語(yǔ)言實(shí)現(xiàn)騎士飛行棋小游戲

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)騎士飛行棋小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-02-02
  • 一文帶你了解C語(yǔ)言中static關(guān)鍵字的3個(gè)作用

    一文帶你了解C語(yǔ)言中static關(guān)鍵字的3個(gè)作用

    static這個(gè)關(guān)鍵字是“靜態(tài)”的意思,在C語(yǔ)言里主要有3個(gè)作用。這篇文章主要通過(guò)一些簡(jiǎn)單示例為大家詳細(xì)講講這3個(gè)左右,感興趣的小伙伴可以了解一下
    2023-04-04
  • 原創(chuàng)的C語(yǔ)言控制臺(tái)小游戲

    原創(chuàng)的C語(yǔ)言控制臺(tái)小游戲

    本文給大家分享的是個(gè)人原創(chuàng)設(shè)計(jì)的一個(gè)C語(yǔ)言控制臺(tái)小游戲,非常的簡(jiǎn)單,但是挺好玩的,推薦給大家,有需要的小伙伴也可以自由擴(kuò)展下。
    2015-03-03
  • MFC實(shí)現(xiàn)簡(jiǎn)單計(jì)算器

    MFC實(shí)現(xiàn)簡(jiǎn)單計(jì)算器

    這篇文章主要為大家詳細(xì)介紹了MFC實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-05-05
  • C++?Boost?ProgramOptions超詳細(xì)講解

    C++?Boost?ProgramOptions超詳細(xì)講解

    Boost是為C++語(yǔ)言標(biāo)準(zhǔn)庫(kù)提供擴(kuò)展的一些C++程序庫(kù)的總稱(chēng)。Boost庫(kù)是一個(gè)可移植、提供源代碼的C++庫(kù),作為標(biāo)準(zhǔn)庫(kù)的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開(kāi)發(fā)引擎之一,是為C++語(yǔ)言標(biāo)準(zhǔn)庫(kù)提供擴(kuò)展的一些C++程序庫(kù)的總稱(chēng)
    2022-11-11
  • 詳解C++ 中的臨時(shí)對(duì)象

    詳解C++ 中的臨時(shí)對(duì)象

    這篇文章主要介紹了C++ 中的臨時(shí)對(duì)象的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)c++,感興趣的朋友可以了解下
    2020-08-08
  • Objective-C 消息傳遞機(jī)制詳解

    Objective-C 消息傳遞機(jī)制詳解

    Objective-C語(yǔ)言中方法的傳遞有二種:①Selector ② Blocks,本文主要說(shuō)一下Selector,本文以O(shè)bjective-C 消息傳遞機(jī)制進(jìn)行詳細(xì)介紹,關(guān)于Blocks會(huì)在后續(xù)總結(jié)一下
    2012-11-11

最新評(píng)論