解决windows api显示透明gif时概率性闪现背景的问题

背景描述

在之前的博客中已经详细介绍了背景,这里给出实际显示效果图吧。
这里写图片描述
这是需求的核心部分,就是用透明的无边框的窗口显示gif图片。本文的重点并不是介绍如何实现这个功能(如需了解见之前的博客),而是要解决一个更加头疼的问题,在开始显示gif动图之前,窗口概率性显示白色背景。概率性的问题,往往是最头疼的。
在之前的博客中,也尝试了多种思路去解决这种概率性闪现背景的问题,遗憾的是只是降低了出现的概率。

采用新的创建窗口的函数

之前实现这个功能是采用CreateWindow,看见网上有人用CreateWindowEx,就想改用这个会不会就不概率性闪现背景了,于是经过各种痛苦的调试,在之前的代码基础上写出了如下代码:

const int kInstallInterfaceWidth = 523;
const int kInstallInterfaceHeight = 357;

int InstallMainWindow::Execute(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
    WNDCLASSEX wcex = { 0 };
    wcex.cbSize = sizeof(WNDCLASSEX);
    wcex.style = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc = (WNDPROC)WndProc;
    wcex.cbClsExtra = 0;
    wcex.cbWndExtra = 0;
    wcex.hInstance = hInstance;
    wcex.hIcon = NULL;
    wcex.hIconSm = NULL;
    wcex.hCursor = NULL;
    wcex.hbrBackground = CreateSolidBrush(RGB(251, 255, 242));
    wcex.lpszMenuName = NULL;
    // If forget to set this attribute, the program will throw an exception.
    wcex.lpszClassName = kWindowClassName;


    // Register the window
    ::RegisterClassEx(&wcex);

    int xLoc = ((GetSystemMetrics(SM_CXSCREEN) / 2) - kInstallInterfaceWidth / 2); 
    int yLoc = ((GetSystemMetrics(SM_CYSCREEN) / 2) - kInstallInterfaceHeight / 2); 
    DWORD main_window_style = WS_POPUP;
    main_window_ = CreateWindowEx(NULL, kWindowClassName, L"", main_window_style, xLoc, yLoc,
        kInstallInterfaceWidth, kInstallInterfaceHeight, NULL, NULL, hInstance, NULL);

    ::ShowWindow(main_window_, nCmdShow);
    ::UpdateWindow(main_window_);

    // 实现窗口透明必须设置的属性
    LONG ret = ::GetWindowLongPtr(main_window_, GWL_EXSTYLE);
    ret = ret | WS_EX_LAYERED;
    ::SetWindowLongPtr(main_window_, GWL_EXSTYLE, ret);

    // 实现窗口置顶
    ::SetWindowPos(main_window_, HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE);
    SetForegroundWindow(main_window_);
    SetFocus(main_window_);

    MSG msg;
    while (::GetMessage(&msg, 0, 0, 0))
    {            
        ::TranslateMessage(&msg);   // Translate keyboard message
        ::DispatchMessage(&msg);    // Dispatch message to the corresponding window
    }

    return (int)msg.wParam;
}

LRESULT CALLBACK InstallMainWindow::WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    switch (message)
    {
    case WM_PAINT:
    {
        ::SetLayeredWindowAttributes(hWnd, RGB(251, 255, 242), 0, LWA_COLORKEY);
        static bool first_gif_flag = true;
        if (first_gif_flag) {
            first_gif_flag = false;   // To avoid too many threads.

            std::thread gif_thread(showimage, hWnd);
            gif_thread.detach();
        }
        break;
    }
    case WM_DESTROY:
        ::PostQuitMessage(0);
        break;

    }

    return ::DefWindowProc(hWnd, message, wParam, lParam);
}

int main()
{
    HINSTANCE h_instance = GetModuleHandle(NULL);
    install_interface::InstallMainWindow& main_window = install_interface::InstallMainWindow::GetMainWindow();
    main_window.Execute(h_instance, h_instance, NULL, SW_SHOWNORMAL);
    return 0;
}

想了解showimage的实现,参考:win32双缓冲实现gif图片的动态显示
这段代码相对之前的代码,关键区别在于创建的窗口的样式改变了,窗口样式改为WS_POPUP,这种样式的窗口在去除窗口边框时更加方便,不用再像之前那样需要在WM_SIZE消息的响应中来实现。
在经过一番调试之后,运行了几次发现在开始显示gif之前没有闪现背景了,但我知道没有这么简单,大概连续运行了二三十次之后还是闪现了背景,运行了六十次闪现了两次背景,相比之前的方式已经有了改进,但是代码仍有瑕疵。

继续改进

在找到合适的解决方案之前,我做出了几乎能够想到的所有修改尝试,包括增加窗口的样式,或者减少样式,修改wcex.hIcon等类似属性的值,以及各种调整代码的位置,遗憾的是都没有实际效果。
怀着绝望的心情,在网上肆意浏览windows api创建窗口相关的几乎所有中文内容,希望能够找到蛛丝马迹。无意间看到空画刷这个词,就想到是否可以用空画刷来创建窗口,但是担心用空画刷创建的窗口,能否正常显示gif图片,毕竟之前经常出现修改代码后连基本的显示gif都失效了,但还是要尝试一下。在一番搜索之后,终于看到有人用空画刷创建窗口,于是修改前面的一行代码,修改如下:

wcex.hbrBackground = (HBRUSH)GetStockObject(NULL_BRUSH);

终于,经过这样设置之后,运行时再也没有闪现窗口了,连续运行代码很多次都没有出现这个问题了。
至此,这个问题终于解决了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值