在项目开发中遇到这样的需求:就是要求整个窗口不透明,但是其中某个区域透明(如果是窗口全透明,那比较简单,如果是窗口四周透明,也比较简单,但是要求窗口中间某个区域透明,这就有点麻烦了),下面分别记录Win32窗口和Qt窗口实现不透明窗口中间某个区域透明的方法,其实都比较简单。
首先说Win32窗口,效果图如下所示:
废话不多说,直接上代码:
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
static HDC hDC = GetWindowDC(hWnd);
switch (message)
{
case WM_COMMAND:
{
int wmId = LOWORD(wParam);
// 分析菜单选择:
switch (wmId)
{
case IDM_EXIT:
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
}
break;
case WM_PAINT:
{
PAINTSTRUCT ps;
HDC hdc = BeginPaint(hWnd, &ps);
// TODO: 在此处添加使用 hdc 的任何绘图代码...
EndPaint(hWnd, &ps);
}
break;
case WM_ERASEBKGND:
{
break;
}
case WM_CREATE:
{
RECT rect;
GetWindowRect(hWnd, &rect);
HRGN hRgn1 = CreateRectRgn(0, 0, rect.right - rect.left, rect.bottom - rect.top);
HRGN hRgn2 = CreateRectRgn(10, 50, 200, 250);
CombineRgn(hRgn1, hRgn1, hRgn2, RGN_XOR);
SetWindowRgn(hWnd, hRgn1, TRUE);
DeleteObject(hRgn1);
DeleteObject(hRgn2);
break;
}
case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}
这里在创建窗口的时候(WM_CREATE),使用SetWindowRgn函数将一个矩形区域设置为透明,原理不多解释,根据代码可自行理解。
再来说Qt窗口实现效果,还是先上效果图:
这里暂时不知道为什么设置了Mask后,窗口标题那些也不在了,不过,本文重点是窗口部分透明,下面是实现代码:
void Widget::SetShelter(QWidget* pWidget, QWidget* pBaseWidget, bool bShelter, const QVector<QRect>& vecShelterRect)
{
pWidget->clearMask();
if (bShelter)
{
Qt::WindowFlags flags = pWidget->windowFlags();
if (!flags.testFlag(Qt::Window))
{
QRegion rgWhole(pWidget->rect());
for (int i = 0; i < vecShelterRect.size(); i++)
{
rgWhole = rgWhole.xored(vecShelterRect[i]);
}
pWidget->setMask(rgWhole);
}
else
{
QPoint ptBase = pBaseWidget->mapFromGlobal(pWidget->geometry().topLeft());
QRegion rgWidget(QRect(ptBase, pWidget->size()));
for (int i = 0; i < vecShelterRect.size(); i++)
{
rgWidget = rgWidget.xored(vecShelterRect[i]);
}
rgWidget.translate(-ptBase.x(), -ptBase.y());
pWidget->setMask(rgWidget);
}
}
}
这里是需要判断一下窗口是不是弹出窗口,是弹出窗口的话,需要重新计算坐标的,调用方法也挺简单的,如下所示:
QVector<QRect> vecShelterRect;
vecShelterRect.append(QRect(10,10,300,300));
SetShelter(this,this,true,vecShelterRect);
好啦,Win32窗口和Qt窗口部分区域透明的实现就总结到这里,扩展一下就可以应用于MFC窗口、QML窗口、PyQt窗口上,方法都是一样的,这里就不多写了。