mirror of
https://github.com/HIllya51/LunaTranslator.git
synced 2025-01-01 10:04:12 +08:00
182 lines
6.4 KiB
C++
182 lines
6.4 KiB
C++
|
|
#include "define.h"
|
|
|
|
void GetVirtualDesktopRect(RECT &rect)
|
|
{
|
|
// 获取虚拟桌面的尺寸和位置
|
|
rect.left = GetSystemMetrics(SM_XVIRTUALSCREEN);
|
|
rect.top = GetSystemMetrics(SM_YVIRTUALSCREEN);
|
|
rect.right = rect.left + GetSystemMetrics(SM_CXVIRTUALSCREEN);
|
|
rect.bottom = rect.top + GetSystemMetrics(SM_CYVIRTUALSCREEN);
|
|
}
|
|
|
|
HBITMAP GetBitmap(RECT &rect, HDC hDC)
|
|
{
|
|
HDC hMemDC;
|
|
int x, y;
|
|
int nWidth, nHeight;
|
|
HBITMAP hBitmap, hOldBitmap;
|
|
|
|
hMemDC = CreateCompatibleDC(hDC);
|
|
// RECT rect;
|
|
// GetVirtualDesktopRect(rect);
|
|
hBitmap = CreateCompatibleBitmap(hDC, rect.right - rect.left, rect.bottom - rect.top);
|
|
hOldBitmap = (HBITMAP)SelectObject(hMemDC, hBitmap);
|
|
|
|
BitBlt(hMemDC, 0, 0, rect.right - rect.left, rect.bottom - rect.top, hDC, rect.left, rect.top, SRCCOPY);
|
|
|
|
hBitmap = (HBITMAP)SelectObject(hMemDC, hOldBitmap);
|
|
|
|
DeleteDC(hMemDC);
|
|
|
|
return hBitmap;
|
|
}
|
|
int SaveBitmapToFile(HBITMAP hBitmap, LPCWSTR lpFileName)
|
|
{
|
|
WORD wBitCount; // 位图中每个像素所占字节数
|
|
// 定义调色板大小,位图中像素字节大小,位图文件大小,写入文件字节数
|
|
DWORD dwPaletteSize = 0, dwBmBitsSize, dwDIBSize, dwWritten;
|
|
BITMAP Bitmap; // 位图属性结构
|
|
BITMAPFILEHEADER bmfHdr; // 位图文件头结构
|
|
BITMAPINFOHEADER bi; // 位图信息头结构
|
|
HANDLE fh; // 定义文件,分配内存句柄,调色板句柄
|
|
LPSTR lpbk, lpmem;
|
|
|
|
wBitCount = 32;
|
|
// 设置位图信息头结构
|
|
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;
|
|
|
|
// 创建位图文件
|
|
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) + dwBmBitsSize;
|
|
bmfHdr.bfSize = dwDIBSize;
|
|
bmfHdr.bfReserved1 = 0;
|
|
bmfHdr.bfReserved2 = 0;
|
|
bmfHdr.bfOffBits = (DWORD)sizeof(BITMAPFILEHEADER) + (DWORD)sizeof(BITMAPINFOHEADER);
|
|
|
|
// 写入位图文件头
|
|
WriteFile(fh, (LPSTR)&bmfHdr, sizeof(BITMAPFILEHEADER), &dwWritten, NULL);
|
|
// 写入位图信息头
|
|
WriteFile(fh, (LPSTR)&bi, sizeof(BITMAPINFOHEADER), &dwWritten, NULL);
|
|
|
|
// 获取位图阵列
|
|
lpmem = new char[dwBmBitsSize];
|
|
lpbk = (LPSTR) new char[dwBmBitsSize];
|
|
GetBitmapBits(hBitmap, dwBmBitsSize, lpmem); // 正向的内存图象数据
|
|
|
|
// 转化为倒向数据(仅在bmHeight为正时需要)
|
|
for (int i = 0; i < Bitmap.bmHeight; i++)
|
|
{
|
|
memcpy(lpbk + Bitmap.bmWidth * i * 4, lpmem + Bitmap.bmWidth * (Bitmap.bmHeight - i - 1) * 4, Bitmap.bmWidth * 4);
|
|
}
|
|
// 写位图数据
|
|
WriteFile(fh, lpbk, dwBmBitsSize, &dwWritten, NULL);
|
|
|
|
// 清除
|
|
delete[] lpbk;
|
|
delete[] lpmem;
|
|
|
|
CloseHandle(fh);
|
|
return TRUE;
|
|
}
|
|
|
|
BYTE *SaveBitmapToBuffer(HBITMAP hBitmap, size_t *size)
|
|
{
|
|
WORD wBitCount; // 位图中每个像素所占字节数
|
|
// 定义调色板大小,位图中像素字节大小,位图文件大小,写入文件字节数
|
|
DWORD dwPaletteSize = 0, dwBmBitsSize, dwDIBSize, dwWritten;
|
|
BITMAP Bitmap; // 位图属性结构
|
|
BITMAPFILEHEADER bmfHdr; // 位图文件头结构
|
|
BITMAPINFOHEADER bi; // 位图信息头结构
|
|
|
|
LPSTR lpbk, lpmem;
|
|
|
|
wBitCount = 32;
|
|
// 设置位图信息头结构
|
|
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;
|
|
|
|
// 设置位图文件头
|
|
bmfHdr.bfType = 0x4D42; // "BM"
|
|
dwDIBSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + dwBmBitsSize;
|
|
bmfHdr.bfSize = dwDIBSize;
|
|
bmfHdr.bfReserved1 = 0;
|
|
bmfHdr.bfReserved2 = 0;
|
|
bmfHdr.bfOffBits = (DWORD)sizeof(BITMAPFILEHEADER) + (DWORD)sizeof(BITMAPINFOHEADER);
|
|
auto buffer = new BYTE[sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + dwBmBitsSize];
|
|
// 写入位图文件头
|
|
// WriteFile(fh, (LPSTR)&bmfHdr, sizeof(BITMAPFILEHEADER), &dwWritten, NULL);
|
|
// 写入位图信息头
|
|
// WriteFile(fh, (LPSTR)&bi, sizeof(BITMAPINFOHEADER), &dwWritten, NULL);
|
|
|
|
memcpy(buffer, &bmfHdr, sizeof(BITMAPFILEHEADER));
|
|
memcpy(buffer + sizeof(BITMAPFILEHEADER), &bi, sizeof(BITMAPINFOHEADER));
|
|
// 获取位图阵列
|
|
lpmem = new char[dwBmBitsSize];
|
|
GetBitmapBits(hBitmap, dwBmBitsSize, lpmem); // 正向的内存图象数据
|
|
|
|
for (int i = 0; i < Bitmap.bmHeight; i++)
|
|
{
|
|
memcpy(buffer + sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + Bitmap.bmWidth * i * 4, lpmem + Bitmap.bmWidth * (Bitmap.bmHeight - i - 1) * 4, Bitmap.bmWidth * 4);
|
|
}
|
|
// 写位图数据
|
|
// WriteFile(fh, lpbk, dwBmBitsSize, &dwWritten, NULL);
|
|
|
|
*size = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + dwBmBitsSize;
|
|
// 清除
|
|
delete[] lpmem;
|
|
|
|
return buffer;
|
|
}
|
|
DECLARE BYTE *gdi_screenshot(HWND hwnd, RECT rect, size_t *size)
|
|
{
|
|
*size = 0;
|
|
if (rect.bottom == rect.top || rect.left == rect.right)
|
|
return nullptr;
|
|
if (!hwnd)
|
|
hwnd = GetDesktopWindow();
|
|
auto hdc = GetDC(hwnd);
|
|
if (!hdc)
|
|
return nullptr;
|
|
auto bm = GetBitmap(rect, hdc);
|
|
// SaveBitmapToFile(bm, LR"(.\2.bmp)");
|
|
auto bf = SaveBitmapToBuffer(bm, size);
|
|
DeleteObject(bm);
|
|
ReleaseDC(hwnd, hdc);
|
|
return bf;
|
|
}
|
|
|
|
DECLARE void maximum_window(HWND hwnd)
|
|
{
|
|
RECT rect;
|
|
GetVirtualDesktopRect(rect);
|
|
MoveWindow(hwnd, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, TRUE);
|
|
} |