質問

AppBar API を使用して画面の上部にアプリケーション バーとしてインストールする Windows アプリがあります (Windows タスク バー自体に似ています)。これはうまく機能し、それに応じてデスクトップのサイズが調整されるため、アプリケーションは常に表示されます。

ただし、ユーザーが「デスクトップの表示」(Windows+D) を選択すると、アプリケーションは非表示になります。アプリケーションを確実に表示できるように「デスクトップの表示」をトラップする方法を知っている人はいますか (Windows はすべてのトップレベル ウィンドウを列挙し、ShowWindow(SW_HIDE) で非表示にすると仮定しています。)

役に立ちましたか?

解決

次のコードを使用して、フォーム負荷しながら関数にウィンドウハンドルを渡します。 うまくいけば、これはあなたの問題を解決します。

public void SetFormOnDesktop(IntPtr hwnd)
{
    IntPtr hwndf = hwnd;
    IntPtr hwndParent = FindWindow("ProgMan", null);
    SetParent(hwndf, hwndParent);
}

他のヒント

私はウィンドウを最上位のウィンドウとして設定しているという印象を受けました(経由 ウィンドウ位置の設定 および HWND_TOPMOST フラグ) により、デスクトップがそれを覆うことができなくなりました。Windows + D では、すべてのウィンドウを最小化してから、Z オーダーでデスクトップを上げて最小化できないウィンドウを隠します (まあ、 とにかくワンポイント)。CreateWindowEx に WS_MINIMIZEBOX を渡さない、または WS_EX_TOOLWINDOW を使用することで、ウィンドウを最小化できないようにできると信じていますが、その点については 100% 保証しているわけではありません。

より複雑なアプローチは、SetWindowsHookEx と KeyboardProc を使用してグローバル キーボードをフックすることです。これはユーザー エクスペリエンスに悪影響を及ぼします。


私は、私が話していることの非常に単純な例をコード化してみました。次のコードは、最小化されていないウィンドウ、またはユーザーが Windows + D キーを押しても覆われていないウィンドウを作成します。Windows 7 では、デスクトップ上のガジェットをその上に移動できることに注意してください。それは本当に説明できません。

#include <windows.h>
#include <tchar.h>

#define WIN_TITLE _T("Resists Win+D Window")
#define WIN_CLASS _T("Resists Win+D Class")

LRESULT CALLBACK CustomWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
    //Special behavior goes here

    return DefWindowProc(hwnd, uMsg, wParam, lParam);
}

HWND CreateMainWindow(HINSTANCE hInstance)
{
    WNDCLASSEX wcex;

    wcex.cbSize = sizeof(WNDCLASSEX);
    wcex.style          = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc    = CustomWndProc;
    wcex.cbClsExtra     = 0;
    wcex.cbWndExtra     = 0;
    wcex.hInstance      = hInstance;
    wcex.hIcon          = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_APPLICATION));
    wcex.hCursor        = LoadCursor(NULL, IDC_ARROW);
    wcex.hbrBackground  = (HBRUSH)(COLOR_WINDOW+1);
    wcex.lpszMenuName   = NULL;
    wcex.lpszClassName  = WIN_CLASS;
    wcex.hIconSm        = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_APPLICATION));

    if (!RegisterClassEx(&wcex))
    {
        exit(1);
    }

    HWND hWnd = CreateWindowEx(
        WS_EX_TOOLWINDOW,
        WIN_CLASS,
        WIN_TITLE,
        WS_OVERLAPPEDWINDOW,
        CW_USEDEFAULT,
        CW_USEDEFAULT,
        500,
        100,
        NULL,
        NULL,
        hInstance,
        NULL
    );

    if (!hWnd)
    {
        exit(1);
    }

    return hWnd;
}

/*
  Main entry point
*/
int WINAPI WinMain(HINSTANCE hInstance,
                   HINSTANCE hPrevInstance,
                   LPSTR lpCmdLine,
                   int nCmdShow)
{
    HWND hwnd = CreateMainWindow(hInstance);

    ShowWindow(hwnd, nCmdShow);
    SetWindowPos(hwnd, HWND_TOPMOST, -1, -1, -1, -1, SWP_NOMOVE | SWP_NOSIZE);
    UpdateWindow(hwnd);

    MSG msg;
    while (GetMessage(&msg, NULL, 0, 0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }

    return (int) msg.wParam;
}

あなたのABN_FULLSCREENAPP通知では、作業領域を占有しているウィンドウがデスクトップであるかどうかを判断する必要があり、その場合、ABN_FULLSCREENAPPメッセージを無視します。

P.S。代替実装として、href="http://www.ssware.com/shlobj/shlobj.htm" rel="nofollow noreferrer"> ShellAppBar のコンポーネントの商業

JKS でのの答えに加えて、ここでは、VB.NETのコードを働いているあなたはすでにあなたのフォームを変換すると仮定するとアプリケーションバーへ。あなたのpに必要な/関数FindWindowSetFormOnDesktopを起動します。

'In your form
Public Sub New()
    'Stuff
    SetFormOnDesktop(Me.Handle)
    'More stuff
End Sub

'In your form or somewhere else.
<DllImport("user32.dll", SetLastError:=True, CharSet:=CharSet.Auto)> _
Private Shared Function FindWindow( _
 ByVal lpClassName As String, _
 ByVal lpWindowName As String) As IntPtr
End Function

<DllImport("user32.dll", SetLastError:=True, CharSet:=CharSet.Auto)> _
Public Shared Function SetParent(_
 ByVal hWndChild As IntPtr, ByVal hWndNewParent As IntPtr) As IntPtr
End Function

Public Sub SetFormOnDesktop(hwnd As IntPtr)
    Dim hwndf As IntPtr = hwnd
    Dim hwndParent As IntPtr = FindWindow("ProgMan", Nothing)
    SetParent(hwndf, hwndParent)
End Sub
ライセンス: CC-BY-SA帰属
所属していません StackOverflow
scroll top