在写windows桌面程序时,有时候会出现任务栏上,鼠标滑上去,显示多个窗口缩略图的情形,如下图所示:
通过下面的任务管理器也能看出
现在剖析下,这种情况是如何产生的,经过写win32桌面程序发现,只要窗口有父窗口或者有所有者窗口,则该窗口不会被任务栏单独列出来,此外,具有WS_EX_TOOLWINDOW样式的窗口也不会被单列出来。
对于父窗口和所有者窗口,一定要先弄清楚,为此之前写了一篇博客,用户可以根据需要查看。
父窗口和所有者窗口
如下图所示,是一个win32 OVERLAPPED的窗口,上面有个创建窗口,点击此按钮,会创建一个新的窗口,此窗口在创建的时候,若没有指明hWndParent,且没有用SetParent为其设置父窗口,且创建的时候,没有指明WS_EX_TOOLWINDOW扩展样式,则该新建的窗口被单列出来。
此时点击按钮时,对应的代码如下:
HWND hWnd = ::CreateWindowExW(0, szWindowClass, L"子窗口", WS_POPUP | WS_SYSMENU | WS_MINIMIZEBOX | WS_MAXIMIZEBOX, \
200, 200, 500, 500, NULL, nullptr, hInst, nullptr);
如果CreateWindowExW设置了hWndParent,或者后面通过SetParent设置了父窗口,则该新建的窗口不会被单列出来。
完整的代码如下:
// Win32AppTest.cpp : 定义应用程序的入口点。
//
#include "framework.h"
#include "Win32AppTest.h"
#include "log/log.h"
#define MAX_LOADSTRING 100
#define IDB_ONE 3301
// 全局变量:
HINSTANCE hInst; // 当前实例
WCHAR szTitle[MAX_LOADSTRING]; // 标题栏文本
WCHAR szWindowClass[MAX_LOADSTRING]; // 主窗口类名
HWND g_hWnd = NULL;
// 此代码模块中包含的函数的前向声明:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
DWORD WINAPI ThreadFun(void* param)
{
//下面是要执行的线程代码
HWND hWnd = (HWND)param;
return 0;
}
int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
_In_opt_ HINSTANCE hPrevInstance,
_In_ LPWSTR lpCmdLine,
_In_ int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine);
// TODO: 在此处放置代码。
// 初始化全局字符串
LoadStringW(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadStringW(hInstance, IDC_WIN32APPTEST, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance);
// 执行应用程序初始化:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
}
HACCEL hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_WIN32APPTEST));
MSG msg;
// 主消息循环:
while (GetMessage(&msg, nullptr, 0, 0))
{
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
return (int) msg.wParam;
}
//
// 函数: MyRegisterClass()
//
// 目标: 注册窗口类。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEXW wcex;
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_WIN32APPTEST));
wcex.hCursor = LoadCursor(nullptr, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = NULL;
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));
return RegisterClassExW(&wcex);
}
//
// 函数: InitInstance(HINSTANCE, int)
//
// 目标: 保存实例句柄并创建主窗口
//
// 注释:
//
// 在此函数中,我们在全局变量中保存实例句柄并
// 创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
hInst = hInstance; // 将实例句柄存储在全局变量中
/*
HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr);
*/
//HWND hWnd = ::CreateWindowExW(0, szWindowClass, L"12345", WS_VISIBLE | WS_DLGFRAME | WS_SYSMENU | WS_MINIMIZEBOX | WS_MAXIMIZEBOX, 100, 100, 1000, 800, NULL, nullptr, hInstance, nullptr);
HWND hWnd = ::CreateWindowExW(0, szWindowClass, L"12345", WS_DLGFRAME | WS_SYSMENU | WS_MINIMIZEBOX | WS_MAXIMIZEBOX, \
100, 100, 1000, 800, NULL, nullptr, hInstance, nullptr);
if (!hWnd)
{
return FALSE;
}
g_hWnd = hWnd;
ShowWindow(hWnd, SW_SHOWNORMAL);
UpdateWindow(hWnd);
return TRUE;
}
//
// 函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
// 目标: 处理主窗口的消息。
//
// WM_COMMAND - 处理应用程序菜单
// WM_PAINT - 绘制主窗口
// WM_DESTROY - 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_COMMAND:
{
int wmId = LOWORD(wParam);
// 分析菜单选择:
switch (wmId)
{
case IDB_ONE:
{
HWND hWnd = ::CreateWindowExW(0, szWindowClass, L"子窗口", WS_POPUP | WS_SYSMENU | WS_MINIMIZEBOX | WS_MAXIMIZEBOX, \
200, 200, 500, 500, NULL, nullptr, hInst, nullptr);
//SetParent(hWnd, g_hWnd);
ShowWindow(hWnd, SW_SHOWNORMAL);
DWORD err = ::GetLastError();
int i = 0;
i++;
}
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_DESTROY:
PostQuitMessage(0);
break;
case WM_CREATE:
{
CreateWindowW(L"Button", L"创建窗口", WS_VISIBLE | WS_CHILD | BS_PUSHBUTTON,
35, 10, 200, 60, hWnd, (HMENU)IDB_ONE, hInst, NULL);
}
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}