我正在尝试制作一个简单的小工具,允许用户从正常操作切换到禁用所有应用程序消息的模式,他们可以使用鼠标进行一些徒手绘制,然后再次切换模式以保持绘制屏幕,而他们做任何他们想做的其他正常事情。如果我决定,这可以演变成一件好事,您可以通过保存您所做的装饰并稍后加载它们来使用装饰屏幕。

当我开始这个时(半年前,在发现 Windows API 后不久)我只是进行了全局鼠标跟踪,并在 GetDC(NULL) hdc 的任何地方画了一个圆圈。当然,问题是当它下面的任何东西更新时它会消失并且仍然可以通过鼠标消息传递,所以如果我按住桌面上的按钮,例如,它会在整个绘画中放置调整大小的矩形东西。

今天,自从 6 个月前最后一次主要工作以来,终于有了一些空闲时间,我决定重新制作它,看看我是否能实现我想要的。我做了一个透明的、最顶层的、WS_CHILD、分层的、最大化的窗口(基本上屏幕不会改变,但在所有东西的顶部都有一个窗口让消息通过)。接下来,我做了这样的设置,当它处于绘画模式时,它将 alpha 值设置为 1 并让用户绘画。在我这样做之前我没有意识到的是,因为窗口的 alpha 值为 1,所以没有一幅画是可见的。

接下来,我尝试使用 GetDC(NULL),但记得当某些内容更新时会被删除。

现在我只是想用位图和dcs将屏幕重复存储到一个dc,在另一个dc上绘画,然后将其复制回存储屏幕的那个,对未绘制的部分具有透明度,然后复制回到屏幕,但我失去了一点心。这是我的源代码(掩码函数取自 this tutorial )。请告诉我其中一些是否不必要。我肯定使用位图进行双缓冲,但我不太确定我需要它们的位置。

//Global mask since it takes longer to make
HBITMAP mask;

//Window Procedure Start
HDC screenDC; //hdc for entire screen
screenDC = GetDC (NULL); //get DC for screen

HDC memDC = CreateCompatibleDC (screenDC); //create DC for holding the screen+paint
HBITMAP bm = CreateCompatibleBitmap (screenDC, GetSystemMetrics (SM_CXSCREEN), GetSystemMetrics (SM_CYSCREEN)); //create bitmap for memDC

HDC paintDC = CreateCompatibleDC (screenDC); //create DC to paint on
HBITMAP paintBM = CreateCompatibleBitmap (screenDC, GetSystemMetrics (SM_CXSCREEN), GetSystemMetrics (SM_CYSCREEN)); //create bitmap for paintDC

SelectObject (memDC, bm); //select bitmap into memDC
SelectObject (paintDC, paintBM); //select painting bitmap into paintDC
BitBlt (memDC, 0, 0, GetSystemMetrics (SM_CXSCREEN), GetSystemMetrics (SM_CYSCREEN), screenDC, 0, 0, SRCCOPY); //copy screen to memDC
SetBkColor (paintDC, RGB(0,0,0)); //set background of paintDC to black so it's all transparent to start

//WM_CREATE
mask = CreateBitmapMask (bm, RGB(0,0,0)); //create black mask (paint colours are limited 1-255 now)

//painting is done into paintDC

//at end of Window Procedure
SelectObject (paintDC, mask); //select mask into paintDC
BitBlt (memDC, 0, 0, GetSystemMetrics (SM_CXSCREEN), GetSystemMetrics (SM_CYSCREEN), paintDC, 0, 0, SRCAND); //this in combination with the next should make it bitblt with all of the black taken out I thought
SelectObject (paintDC, paintBM); //select bitmaps into DCs
SelectObject (memDC, bm);
BitBlt (memDC, 0, 0, GetSystemMetrics (SM_CXSCREEN), GetSystemMetrics (SM_CYSCREEN), paintDC, 0, 0, SRCPAINT); //second part of transparent bitblt
BitBlt (screenDC, 0, 0, GetSystemMetrics (SM_CXSCREEN), GetSystemMetrics (SM_CYSCREEN), paintDC, 0, 0, SRCCOPY); //copy memDC back to screen

DeleteObject (paintBM); //delete stuff
DeleteObject (mask);
DeleteDC (memDC);
DeleteDC (paintDC);
ReleaseDC (hwnd, screenDC);

//CreateBitmapMask() (taken directly from http://www.winprog.org/tutorial/transparency.html
HBITMAP CreateBitmapMask(HBITMAP hbmColour, COLORREF crTransparent)
{
HDC hdcMem, hdcMem2;
HBITMAP hbmMask;
BITMAP bm;

// Create monochrome (1 bit) mask bitmap.

GetObject(hbmColour, sizeof(BITMAP), &bm);
hbmMask = CreateBitmap(bm.bmWidth, bm.bmHeight, 1, 1, NULL);

// Get some HDCs that are compatible with the display driver

hdcMem = CreateCompatibleDC(0);
hdcMem2 = CreateCompatibleDC(0);

SelectObject(hdcMem, hbmColour);
SelectObject(hdcMem2, hbmMask);

// Set the background colour of the colour image to the colour
// you want to be transparent.
SetBkColor(hdcMem, crTransparent);

// Copy the bits from the colour image to the B+W mask... everything
// with the background colour ends up white while everythig else ends up
// black...Just what we wanted.

BitBlt(hdcMem2, 0, 0, bm.bmWidth, bm.bmHeight, hdcMem, 0, 0, SRCCOPY);

// Take our new mask and use it to turn the transparent colour in our
// original colour image to black so the transparency effect will
// work right.
BitBlt(hdcMem, 0, 0, bm.bmWidth, bm.bmHeight, hdcMem2, 0, 0, SRCINVERT);

// Clean up.

DeleteDC(hdcMem);
DeleteDC(hdcMem2);

return hbmMask;
}

我知道代码很可能很糟糕。我正在考虑所有建议,只是我对这个主题不太清楚,并且没有了解正在发生的一切,这使得很难解决。这段代码通过几乎每隔一段时间放置一个全屏黑色矩形来为我运行。

我的主要问题是:有什么方法可以在屏幕上进行绘画而不会在下面的窗口更新时被擦除?我现在能想到的唯一真实的事情是存储用户绘制的细线段的所有位置,并继续在屏幕顶部重新绘制它们。乍一看,它似乎非常低效且浪费内存。

此外,我在编写本文时非常确定,在提供的代码段之前,我不需要任何用于理论内容的代码示例。现在大部分都消失了,但这更像是一个理论问题。

编辑:
我刚刚发现 TransparentBlt 函数似乎非常适合这种情况,所以我尝试使用它代替 SRCPAINT 和 SRCAND BitBlts,它产生了相同的结果:覆盖屏幕的黑色矩形,有时当我的鼠标移过时部分消失事物。

最佳答案

最简单的方法,也许:

在非绘图模式下,使用 SetLayeredWindowAttributes 为透明窗口设置“透明键”颜色。使窗口的 alpha 完全不透明,但使用该键颜色填充窗口(FillRect 或类似颜色),它会全部显示为透明。然后,您以非关键颜色绘制的任何内容都将显示为纯色,位于透明分层窗口下方的所有窗口之上。

要进入绘图模式,一种方法是创建一个新窗口,并在透明层的正下方使用捕获的桌面位图。或者避免使用位图,并使其略微不透明并且全部为纯色 - 例如,看起来桌面是“变灰的”。关键是这个窗口不是完全透明的,所以它能够接收鼠标输入,然后你可以用它在实际的透明层上绘制。

关于c++ - 全屏手绘,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/7604962/

10-09 06:26