背景描述
在之前的博客中已经详细介绍了背景,这里给出实际显示效果图吧。
这是需求的核心部分,就是用透明的无边框的窗口显示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);
终于,经过这样设置之后,运行时再也没有闪现窗口了,连续运行代码很多次都没有出现这个问题了。
至此,这个问题终于解决了。