WS_EX_TOOLWINDOWがウィンドウ破棄でバックグラウンドにジャンプする
質問
奇妙な問題:
- 大きなメモ帳ウィンドウを開く
- ツールウィンドウの作成(スタイルWS_EX_TOOLWINDOW)
- さらに2つのウィンドウを作成(通常のオーバーラップ)(WS_OVERLAPPED)
- これら2つのオーバーラップウィンドウ(デスクトップまたはツールウィンドウの子)を閉じます
- ツールウィンドウはメモ帳ウィンドウの後ろにジャンプします
これが事実である理由を誰もが知っていますか?または私が間違っている可能性がありますか? 「Windowsのバグ」と言いますが、そうなることはほとんどありません。
質問に答えるには:
これはダイアログウィンドウではなく、完全なウィンドウです。正しい子(つまり、デスクトップの子ではない)を作成すると、子のタスクバーエントリは表示されません(おそらく簡単に修正可能)が、いずれにしても、バグは発生します。
問題を示すサンプルコードを含めました。間違ったウィンドウを作成したり、応答していないメッセージに応答するために必要なウィンドウを作成したりすることを望んでいます。
この例では、ツールウィンドウが開きます(必要なタスクバーエントリはありません)。次に、そのウィンドウをクリックすると、サブウィンドウが開きます。サブウィンドウをクリックすると、別のウィンドウが開きます。次に、新しいサブウィンドウと元のウィンドウの両方を閉じます。フォーカスを取得する代わりに、すぐに他のウィンドウ(メモ帳など)の後ろにジャンプします。
ご協力ありがとうございます!
明確にするためのコード例:
// WindowToback.cpp : Defines the entry point for the application.
//
#include "stdafx.h"
#include "WindowToback.h"
// Global Variables:
HINSTANCE g_instance;
HWND g_mainWnd = NULL;
wchar_t *szWindowClass = L"WindowToBackSub";
wchar_t *szWindowClass2 = L"WindowToBackSub2";
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK WndProc2(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM);
int APIENTRY _tWinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine);
MSG msg;
MyRegisterClass(hInstance);
// Perform application initialization:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
}
// Main message loop:
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return (int) msg.wParam;
}
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEX 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_WINDOWTOBACK));
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = MAKEINTRESOURCE(IDC_WINDOWTOBACK);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));
RegisterClassEx(&wcex);
wcex.lpfnWndProc = WndProc2;
wcex.lpszClassName = szWindowClass2;
return RegisterClassEx(&wcex);
}
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
g_instance = hInstance;
g_mainWnd = CreateWindowEx(WS_EX_TOOLWINDOW,szWindowClass, szWindowClass,WS_OVERLAPPED,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
if (!g_mainWnd) return FALSE;
ShowWindow(g_mainWnd, nCmdShow);
UpdateWindow(g_mainWnd);
return TRUE;
}
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_LBUTTONDOWN:
{
HWND l_hwnd = CreateWindow(szWindowClass2, szWindowClass2, WS_VISIBLE | WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, g_instance, NULL);
ShowWindow(l_hwnd,SW_SHOW);
break;
}
case WM_DESTROY:
{
PostQuitMessage(0);
return 0;
}
}
return DefWindowProc(hWnd, message, wParam, lParam);
}
LRESULT CALLBACK WndProc2(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_LBUTTONDOWN:
{
HWND l_hwnd = CreateWindow(szWindowClass2, szWindowClass2, WS_VISIBLE | WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, g_instance, NULL);
ShowWindow(l_hwnd,SW_SHOW);
}
break;
}
return DefWindowProc(hWnd, message, wParam, lParam);
}
解決
これは驚くことではありません。実際、それはまさに私が期待する動作です。
ツールウィンドウは下にジャンプしていません。むしろメモ帳が跳ね上がります。
アクティブ化されたウィンドウを閉じました。システムは、zオーダーで次に高い最上位ウィンドウをアクティブにします。この点では、ツールウィンドウはトップレベルウィンドウとしてカウントされません(これはツールウィンドウの意味の一部です)。そのため、メモ帳がアクティブになり、一番上に表示されます。
代わりにツールウィンドウをアクティブにしたい場合、おそらくツールウィンドウは本当に必要ないでしょう。
他のヒント
3つのウィンドウダイアログは別のメインウィンドウにありますか、それともアプリケーションそのものですか?
ダイアログウィンドウの場合、親ウィンドウが正しく割り当てられていることを確認します。
それらがアプリケーションウィンドウの場合、タスクバーに表示されていることを確認します。
問題に関する詳細情報がなければ、より意味のある答えを出すことは困難です。