win32透明窗口

一、It is not quite trivial to draw semitransparent translucent windows. The algorithm follows:

  1. Create a WS_POPUP window with the WS_EX_LAYERED style.
  2. Create a 32 bit DIB Section using the CreateDIBSection() function and select it to any compatible DC. It will be an off-screen plain to render window contents to.
  3. Render the window contents, preserving the alpha channel.
  4. Call the UpdateLayeredWindow() function to draw the window to the screen.

二、

https://stackoverflow.com/questions/3970066/creating-a-transparent-window-in-c-win32winapi - Creating a transparent window in C++ Win32 - Stack Overflow

Creating a transparent window in C++ Win32

I'm creating what should be a very simple Win32 C++ app whose sole purpose it to ONLY display a semi-transparent PNG. The window shouldn't have any chrome, and all the opacity should be controlled in the PNG itself.

My problem is that the window doesn't repaint when the content under the window changes, so the transparent areas of the PNG are "stuck" with what was under the window when the application was initially started.

Here's the line where I setup the new window:

hWnd = CreateWindowEx(WS_EX_TOPMOST, szWindowClass, szTitle, WS_POPUP, 0, height/2 - 20, 40, 102, NULL, NULL, hInstance, 0);

 

For the call to RegisterClassEx, I have this set for the background:

wcex.hbrBackground = (HBRUSH)0;

Here is my handler for WM_PAINT message:

 case WM_PAINT:
 {
   hdc = BeginPaint(hWnd, &ps);
   Gdiplus::Graphics graphics(hdc);
   graphics.DrawImage(*m_pBitmap, 0, 0);
   EndPaint(hWnd, &ps);
   break;
 }

One thing to note is that the application is always docked to the left of the screen and doesn't move. But, what's underneath the application may change as the user opens, closes or moves windows under it.

When the application first starts, it looks perfect. The transparent (and simi-transparent) parts of the PNG show through perfectly. BUT, when the background underneath the application changes, the background DOESN'T update, it just stays the same from when the application first started. In fact, WM_PAINT (or WM_ERASEBKGND does not get called when the background changes).

I've been playing with this for quite a while and have gotten close to getting 100% right, but not quite there. For instance, I've tried setting the background to (HBRUSH) NULL_BRUSH and I've tried handling WM_ERASEBKGND.

What can be done to get the window to repaint when the contents under it changes?

 SetBKMode and SetBKColor are the APIs I have used to make the transparent parent control. 

2 Answers

I was able to do exactly what I wanted by using the code from Part 1 and Part 2 of this series:

Displaying a Splash Screen with C++


Those blog posts are talking about displaying a splash screen in Win32 C++, but it was almost identical to what I needed to do. I believe the part that I was missing was that instead of just painting the PNG to the window using GDI+, I needed to use the UpdateLayeredWindow function with the proper BLENDFUNCTION parameter. I'll paste the SetSplashImage method below, which can be found in Part 2 in the link above:

void SetSplashImage(HWND hwndSplash, HBITMAP hbmpSplash)
{
  // get the size of the bitmap
  BITMAP bm;
  GetObject(hbmpSplash, sizeof(bm), &bm);
  SIZE sizeSplash = { bm.bmWidth, bm.bmHeight };

  // get the primary monitor's info
  POINT ptZero = { 0 };
  HMONITOR hmonPrimary = MonitorFromPoint(ptZero, MONITOR_DEFAULTTOPRIMARY);
  MONITORINFO monitorinfo = { 0 };
  monitorinfo.cbSize = sizeof(monitorinfo);
  GetMonitorInfo(hmonPrimary, &monitorinfo);

  // center the splash screen in the middle of the primary work area
  const RECT & rcWork = monitorinfo.rcWork;
  POINT ptOrigin;
  ptOrigin.x = 0;
  ptOrigin.y = rcWork.top + (rcWork.bottom - rcWork.top - sizeSplash.cy) / 2;

  // create a memory DC holding the splash bitmap
  HDC hdcScreen = GetDC(NULL);
  HDC hdcMem = CreateCompatibleDC(hdcScreen);
  HBITMAP hbmpOld = (HBITMAP) SelectObject(hdcMem, hbmpSplash);

  // use the source image's alpha channel for blending
  BLENDFUNCTION blend = { 0 };
  blend.BlendOp = AC_SRC_OVER;
  blend.SourceConstantAlpha = 255;
  blend.AlphaFormat = AC_SRC_ALPHA;

  // paint the window (in the right location) with the alpha-blended bitmap
  UpdateLayeredWindow(hwndSplash, hdcScreen, &ptOrigin, &sizeSplash,
      hdcMem, &ptZero, RGB(0, 0, 0), &blend, ULW_ALPHA);

  // delete temporary objects
  SelectObject(hdcMem, hbmpOld);
  DeleteDC(hdcMem);
  ReleaseDC(NULL, hdcScreen);
}

Share

Follow

edited Aug 6, 2020 at 15:51

Ian Boyd

234k238238 gold badges835835 silver badges11601160 bronze badges

answered Oct 19, 2010 at 19:03

adoss

1,08111 gold badge99 silver badges99 bronze badges

  • best answer, also support automatic PNG alpha blend when used with WIC (Windows Imaging Component) via COM interface. 

    – dns

     Jul 5, 2015 at 1:51 
  • Ended up wasting a lot of time with AlphaBlend. This solution is proabably one of the easiest ways to achieve Per-Pixel Alpha Blending, specially when you are updating the window. 

    – TheBlueNotebook

     Jan 11, 2016 at 12:30

Add a comment

24

Use the SetLayeredWindowAttributesarchive function, this allows you to set a mask color that will become transparent, thus allowing the background to show through.

You will also need to configure your window with the layered flag, e.g.:

SetWindowLong(hwnd, GWL_EXSTYLE, GetWindowLong(hwnd, GWL_EXSTYLE) | WS_EX_LAYERED);

After that it's fairly simple:

// Make red pixels transparent:
SetLayeredWindowAttributes(hwnd, RGB(255,0,0), 0, LWA_COLORKEY);

When your PNG contains semi-transparent pixels that you want to blend with the background, this becomes more complicated. You could try looking at the approach in this CodeProject article:

Cool, Semi-transparent and Shaped Dialogs with Standard Controls for Windows 2000 and Above

Share

Follow

edited Aug 6, 2020 at 15:50

Ian Boyd

234k238238 gold badges835835 silver badges11601160 bronze badges

answered Oct 19, 2010 at 15:53

Simon Steele

11.5k44 gold badges4343 silver badges6767 bronze badges

  • 1

    I've tried this an it ALMOST works. The issue is that there are some areas of the PNG which are simi-transparent, and the color key method only makes the pixels which are 100% opacity of the color key color value transparent. If all else fails, my fallback will be to remove all the semi-transparent regions, but I'd really rather avoid doing that. 

    – adoss

     Oct 19, 2010 at 16:24
  • You can try more complicated hacks like the one in the article I've linked, but essentially as soon as you're looking to blend a pre-existing image with the desktop contents things start getting complicated. 

    – Simon Steele

     Oct 19, 2010 at 16:45
  • Check out the answer that I posted. I like that code a whole lot better than what I was seeing in this CodeProject article. Not only that, but I believe what I found is the proper solution instead of a hack. Thanks for the help! 

    – adoss

     Oct 19, 2010 at 19:05
  • Ah nice, I hadn't seen the BLENDFUNCTION stuff before - you're welcome! 

    – Simon Steele

     Oct 20, 2010 at 8:54
  • 1

    why don't you set the forth parameter of SetLayeredWindowAttributes to LWA_ALPHA 

    – Valen

     Apr 18, 2021 at 2:21

三、C++ (Cpp) UpdateLayeredWindow示例 - HotExamples

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值