如何使用 bitblt 在屏幕上绘制由位图制作的数组?
How do I draw an array made bitmap on screen using bitblt?
而不是使用Moveto
和LineTo
绘制,我想使用手工制作的位图,我将用我自己创建的数组填充该数组,并使用它来填充屏幕。
现在,数组只是用红色填充, 但是当我画到屏幕上时,我全身都是黑色的。
这是代码:
void CCGWorkView::OnDraw(CDC* pDC)
{
CCGWorkDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
return;
pDCToUse->FillSolidRect(&r, world.bg_color);
BITMAPINFO bminfo;
CPaintDC hdc(this);
CRect rect;
GetWindowRect(&rect);
int h = rect.bottom - rect.top,
w = rect.right - rect.left;
int *bits = new int[w * h];
HDC hdcMem = CreateCompatibleDC(hdc);
HBITMAP bm = CreateCompatibleBitmap(hdc, w, h);
SelectObject(hdcMem, bm);
bminfo.bmiHeader.biSize = sizeof(bminfo.bmiHeader);
bminfo.bmiHeader.biWidth = w;
bminfo.bmiHeader.biHeight = h;
bminfo.bmiHeader.biPlanes = 1;
bminfo.bmiHeader.biBitCount = 32;
bminfo.bmiHeader.biCompression = BI_RGB;
bminfo.bmiHeader.biSizeImage = 0;
bminfo.bmiHeader.biXPelsPerMeter = 1;
bminfo.bmiHeader.biYPelsPerMeter = 1;
bminfo.bmiHeader.biClrUsed = 0;
bminfo.bmiHeader.biClrImportant = 0;
for (int i = 0; i < w * h; i++) {
bits[i] = RGB(255, 0, 0);
}
SetDIBits(hdcMem, bm, 0, h, bits, &bminfo, 0);
BitBlt(hdc, rect.left, rect.top, w, h, hdcMem, rect.left, rect.top, SRCCOPY);
DeleteDC(hdcMem);
DeleteObject(bm);
delete bits;
}
您的代码中存在几个问题。
首先,你不需要CPaintDC
,因为pDC
被传递到你的OnDraw()
函数中。 这实际上可能导致您的黑色显示,因为新创建的 DC 在其中选择了 1 像素黑白位图,并且当您调用CreateCompatibleBitmap()
时,该位图也是单色的。
其次,SetDIBits()
期望RGBQUAD
颜色(https://learn.microsoft.com/en-us/windows/win32/api/wingdi/ns-wingdi-rgbquad(,而不是RGB
。
而且,正如 Constantine Georgiou 所建议的那样,您应该在删除位图之前从 DC 中取消选择位图以避免资源泄漏,即使 MFC 为您处理它也是如此。
这是修改后的代码:
void CMFCApplication1View::OnDraw(CDC* pDC)
{
CMFCApplication1Doc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
return;
//pDCToUse->FillSolidRect(&r, world.bg_color);
BITMAPINFO bminfo;
CRect rect;
GetClientRect(&rect);
int h = rect.bottom - rect.top,
w = rect.right - rect.left;
int* bits = new int[w * h];
HDC hdcMem = CreateCompatibleDC(pDC->m_hDC);
HBITMAP bm = CreateCompatibleBitmap(pDC->m_hDC, w, h);
HGDIOBJ hOld = SelectObject(hdcMem, bm);
bminfo.bmiHeader.biSize = sizeof(bminfo.bmiHeader);
bminfo.bmiHeader.biWidth = w;
bminfo.bmiHeader.biHeight = h;
bminfo.bmiHeader.biPlanes = 1;
bminfo.bmiHeader.biBitCount = 32;
bminfo.bmiHeader.biCompression = BI_RGB;
bminfo.bmiHeader.biSizeImage = 0;
bminfo.bmiHeader.biXPelsPerMeter = 1;
bminfo.bmiHeader.biYPelsPerMeter = 1;
bminfo.bmiHeader.biClrUsed = 0;
bminfo.bmiHeader.biClrImportant = 0;
RGBQUAD red = { 0, 0, 255, 0 };
for (int i = 0; i < w * h; i++) {
bits[i] = *((int*)&red);
}
SetDIBits(hdcMem, bm, 0, h, bits, &bminfo, DIB_RGB_COLORS);
BitBlt(pDC->m_hDC, rect.left, rect.top, w, h, hdcMem, rect.left, rect.top, SRCCOPY);
SelectObject(hdcMem, hOld);
DeleteDC(hdcMem);
DeleteObject(bm);
delete[] bits;
}
相关文章:
- 从两个 4x64 位整数数组中获取取模
- 位图到垫子/2D 数组
- 是否可以在 64 位系统中为 11 位系统中的图形创建位邻接数组
- MediaFoundation,位图数组到mp4
- 采用OpenCV Mat<doube>并转换为12位值数组。
- 将 __m128i?布尔数组中每个布尔字节的低位提取到打包的位图
- 将数组写入位图 c++
- 将超过 64 位的数组转换为基数为 10 的字符串
- Vizualize无符号char *数组作为位图
- 访问位图的字节数组
- 如何仅使用数学方法旋转16x16位图数组的内容(无缩放,只需剪切角)
- 将 8 位整数放入 16 位整数数组的最快方法是什么
- 为C++中的每个单元格创建一个只有 2 位的数组
- 表示位在数组中表示的旋转
- scanf 只能将 3 个字符读取到 5 位字符数组中
- 将 8 位值数组重新解释为标准复数值
- 3 位的数组对齐
- 在64位计算机中分配32位整数数组
- 可能的c++ 2位位域数组
- 2bit位域数组对性能和缓存效率的影响