一个应用程序,任务栏上显示多层窗口

在写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;
}


猜你喜欢

转载自blog.csdn.net/tusong86/article/details/121058296