4

PROGDLG_MODAL モードで IProgressDialog を使用しようとしていますが、2 つの問題が発生しています。

まず、MSDN から収集したものと<shlobj.h>IProgressDialog を説明するヘッダー ファイルのブロック コメントから、作業中のスレッドから直接それを使用し、IProgressDialog は別のスレッドから UI を実行することになっています。あれは、

while (there_is_still_work_to_do()) {
    if (pd->HasUserCancelled())
        break;
    do_more_work();
    completed++;
    pd->SetProgress64(completed, total);
}

応答性の高い進行状況ダイアログを起動して実行するには十分です。

ただし、実際には、メッセージを送り込む必要がありますdo_more_work()。そうしないと、ループが終了するまで進行状況ダイアログが表示されません。「オブジェクトはバックグラウンドスレッドで更新を処理します」と言うとき、MSDNを誤解していますか? わかりましたので、メッセージをポンプする必要がありますが、MSDN の言うことにはまだ混乱しています: IProgressDialog のどのメソッドをワーカー スレッドから実行できますか? これを知っていれば、コードを適切に再構築できます。

ただし、さらに深刻な問題はStopProgressDialog()、ウィンドウがすぐに取り壊されないことです。実際、どちらもありませRelease()ん。モーダルでない状況では、進行状況ダイアログが画面にしばらく残っているように見えます。この特定の問題は以前に対処されています。私はそれに行きます。ただし、今回は進行状況ダイアログを UI モーダル ダイアログとして使用しています。MessageBox()IProgressDialog を解放した直後になどの別のダイアログ ボックス関数を呼び出すと、同じオーナー ウィンドウを持つ 2 つのモーダル ダイアログが表示されます。まだ実行されています。

非モーダルのケースは、 を呼び出した後に進行状況ダイアログを非表示にするだけでStopProgressDialog()、他の人によって解決されました。これはウィンドウを非表示にしますが、モダリティについては何もしません。リンクされた質問の最後にある質問者の仮説に基づいて、私もWM_NULLs afterを投稿して送信してみましたStopProgressDialog()。それもうまくいきませんでした。

最後に、WinEvents フックをセットアップして、ウィンドウが破棄されるのを待ち、破棄されたときにイベント オブジェクトを起動しようとしました。これは機能します。これMessageBox()は、進行状況ダイアログが破棄されるまで発生しません。ただし、これは完全には機能しません。メイン ウィンドウがすぐに再びアクティブになることMessageBox()はなく、メイン ウィンドウのタスクバー アイコンをクリックするまでバックグラウンドに表示されることさえありません。

(それでも、LPARAM を WinEvents フックに渡すことができないため、異なるスレッド間で複数の IProgressDialogs を処理したい場合は、監視するウィンドウ ハンドル (および関連するイベント オブジェクト) のグローバル リストが必要になります。幸いなことに、私の目的ではこれは必要ありません.さらに、上記のすべては、CLSID_ProgressDialog も IOleWindow であるという前提の下にあり、それが変更された場合は...)

この WinEvents の方法で何か問題がありますか? 私はMsgWaitForMultipleObjectsEx()正しく呼び出していないと思いますが、IProgressDialog がモダリティを適切に実装していないことが判明した場合、私は運が悪いと思います:S

以下のサンプル プログラムは、上記のすべてを実行します。Windows 7 64 ビット (64 ビットとしてビルド) 上の Visual Studio 2013 でテストしました。

// 14 december 2015
#define _UNICODE
#define UNICODE
#define STRICT
#define STRICT_TYPED_ITEMIDS
// get Windows version right; right now Windows XP
#define WINVER 0x0501
#define _WIN32_WINNT 0x0501
#define _WIN32_WINDOWS 0x0501       /* according to Microsoft's winperf.h */
#define _WIN32_IE 0x0600            /* according to Microsoft's sdkddkver.h */
#define NTDDI_VERSION 0x05010000    /* according to Microsoft's sdkddkver.h */
#include <windows.h>
#include <shlobj.h>
#include <shlwapi.h>

void chk(HRESULT hr) { if (hr != S_OK) DebugBreak(); }

void doWork(bool pumpMessages)
{
    UINT_PTR timer;
    MSG msg;

    if (!pumpMessages) {
        Sleep(2000);
        return;
    }
    timer = SetTimer(NULL, 20, 2000, NULL);
    while (GetMessageW(&msg, NULL, 0, 0)) {
        if (msg.message == WM_TIMER && msg.hwnd == NULL)
            break;
        TranslateMessage(&msg);
        DispatchMessageW(&msg);
    }
    KillTimer(NULL, timer);
}

HWINEVENTHOOK hook;
HWND dialogWindow;
HANDLE dialogEvent;
void CALLBACK onDialogClosed(HWINEVENTHOOK hWinEventHook, DWORD event, HWND hwnd, LONG idObject, LONG idChild, DWORD dwEventThread, DWORD dwmsEventTime)
{
    if (hwnd == dialogWindow)
        SetEvent(dialogEvent);
}
void waitEvent(void)
{
    MSG msg;
    DWORD ret;

    for (;;) {
        ret = MsgWaitForMultipleObjectsEx(1, &dialogEvent,
            INFINITE, QS_ALLINPUT, MWMO_INPUTAVAILABLE);
        if (ret == WAIT_OBJECT_0)   // event
            break;
        if (GetMessage(&msg, NULL, 0, 0) == 0)
            break;
        TranslateMessage(&msg);
        DispatchMessageW(&msg);
    }
}

void rundialogs(HWND parent, bool pumpMessages, bool tryHide, int tryHideWhat, bool abort)
{
    IProgressDialog *pd;
    IOleWindow *olewin;
    HWND hwnd;
    DWORD process;
    DWORD thread;

    chk(CoCreateInstance(CLSID_ProgressDialog, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&pd)));
    chk(pd->SetTitle(L"Test"));
    chk(pd->StartProgressDialog(parent, NULL,
        PROGDLG_NORMAL | PROGDLG_MODAL | PROGDLG_AUTOTIME | PROGDLG_NOMINIMIZE,
        NULL));

    doWork(pumpMessages);

    chk(pd->Timer(PDTIMER_RESET, NULL));
    for (ULONGLONG i = 0; i < 10; i++) {
        if (pd->HasUserCancelled())
            break;

        doWork(pumpMessages);
        if (i == 5 && abort)
            break;

        chk(pd->SetProgress64(i + 1, 10));
    }

    chk(pd->QueryInterface(IID_PPV_ARGS(&olewin)));
    chk(olewin->GetWindow(&hwnd));
    olewin->Release();

    // set up event hoook before stopping the progress dialog
    // this way it won't get sdestroyed before the hook is set up
    if (tryHide && tryHideWhat == 3) {
        thread = GetWindowThreadProcessId(hwnd, &process);
        dialogWindow = hwnd;
        dialogEvent = CreateEvent(NULL, TRUE, TRUE, NULL);
        ResetEvent(dialogEvent);
        hook = SetWinEventHook(EVENT_OBJECT_DESTROY, EVENT_OBJECT_DESTROY,
            NULL, onDialogClosed,
            process, thread,
            WINEVENT_OUTOFCONTEXT);
    }

    chk(pd->StopProgressDialog());
    if (tryHide)
        switch (tryHideWhat) {
        case 0:     // hide
            ShowWindow(hwnd, SW_HIDE);
            break;
        case 1:     // send WM_NULL
            SendMessageW(hwnd, WM_NULL, 0, 0);
            break;
        case 2:     // post WM_NULL
            PostMessageW(hwnd, WM_NULL, 0, 0);
            break;
        case 3:     // winevents
            waitEvent();
            UnhookWinEvent(hook);
            break;
        }
    pd->Release();

    MessageBoxW(parent,
        L"This should be MODAL to the main window!\n"
        L"But you should see that in reality the main window\n"
        L"is still enabled!",
        L"mainwin",
        MB_OK | MB_ICONINFORMATION);
}

HWND button;
HWND checkbox;
HWND checkbox2;
HWND combobox;
HWND checkbox3;

bool ischecked(HWND hwnd) { return SendMessageW(hwnd, BM_GETCHECK, 0, 0) == BST_CHECKED; }

static LRESULT CALLBACK wndproc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
    if (uMsg == WM_COMMAND && lParam == (LPARAM) button)
        rundialogs(hwnd,
            ischecked(checkbox),
            ischecked(checkbox2),
            (int) SendMessageW(combobox, CB_GETCURSEL, 0, 0),
            ischecked(checkbox3));
    if (uMsg == WM_CLOSE)
        PostQuitMessage(0);
    return DefWindowProcW(hwnd, uMsg, wParam, lParam);
}

int WINAPI wWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPWSTR lpCmdLine, int nCmdShow)
{
    WNDCLASSW wc;
    HWND mainwin;
    MSG msg;

    CoInitialize(NULL);

    ZeroMemory(&wc, sizeof (WNDCLASSW));
    wc.lpszClassName = L"mainwin";
    wc.lpfnWndProc = wndproc;
    wc.hInstance = hInstance;
    wc.hbrBackground = (HBRUSH) (COLOR_BTNFACE + 1);
    RegisterClassW(&wc);

    mainwin = CreateWindowExW(0,
        L"mainwin", L"mainwin",
        WS_OVERLAPPEDWINDOW,
        CW_USEDEFAULT, CW_USEDEFAULT,
        200, 220,
        NULL, NULL, hInstance, NULL);

    button = CreateWindowExW(0,
        L"button", L"Click Me",
        BS_PUSHBUTTON | WS_CHILD | WS_VISIBLE,
        10, 10, 150, 100,
        mainwin, (HMENU) 100, hInstance, NULL);

    checkbox = CreateWindowExW(0,
        L"button", L"Pump Messages",
        BS_CHECKBOX | BS_AUTOCHECKBOX | WS_CHILD | WS_VISIBLE,
        10, 110, 150, 20,
        mainwin, (HMENU) 101, hInstance, NULL);
    checkbox2 = CreateWindowExW(0,
        L"button", L"Try",
        BS_CHECKBOX | BS_AUTOCHECKBOX | WS_CHILD | WS_VISIBLE,
        10, 130, 50, 20,
        mainwin, (HMENU) 101, hInstance, NULL);
    combobox = CreateWindowExW(0,
        L"combobox", L"",
        CBS_DROPDOWNLIST | WS_CHILD | WS_VISIBLE,
        60, 130, 100, 100,
        mainwin, (HMENU) 102, hInstance, NULL);
    SendMessageW(combobox, CB_ADDSTRING, 0, (LPARAM) L"Hide");
    SendMessageW(combobox, CB_ADDSTRING, 0, (LPARAM) L"Send");
    SendMessageW(combobox, CB_ADDSTRING, 0, (LPARAM) L"Post");
    SendMessageW(combobox, CB_ADDSTRING, 0, (LPARAM) L"WinEvents");
    SendMessageW(combobox, CB_SETCURSEL, 0, 0);
    checkbox3 = CreateWindowExW(0,
        L"button", L"Abort Early",
        BS_CHECKBOX | BS_AUTOCHECKBOX | WS_CHILD | WS_VISIBLE,
        10, 150, 150, 20,
        mainwin, (HMENU) 103, hInstance, NULL);

    ShowWindow(mainwin, nCmdShow);
    UpdateWindow(mainwin);

    while (GetMessageW(&msg, NULL, 0, 0)) {
        TranslateMessage(&msg);
        DispatchMessageW(&msg);
    }

    CoUninitialize();
    return 0;
}

更新ええ、さらに調べてみると、メッセージボックスを表示せずに進行状況ダイアログを停止すると、所有者ウィンドウにもフォーカスが戻らないようです! 進行状況ダイアログが単にモダリティを正しく処理していないと推測しています。PROGDLG_MODALフラグを削除すると、すべて正常に動作します。まあ:/モダリティを偽造するか、別のものに切り替える必要があります。

進行状況ダイアログ自体の上にメッセージ ボックスを表示することになり、Windows の将来のバージョンで IOleWindow が削除されないことを願っています。もっと良い方法がない限り?または、モダリティを手動で実行し、 の呼び出し後に進行状況ダイアログをモードレスのままにしておくのがStopProgressDialog()十分でない限り (エラーは最初に報告された後にのみ報告さStopProgressDialog()れます)、もう一度...

4

3 に答える 3

1

これが「クリーンな」解決策かどうかはわかりませんが、その進行状況ダイアログへのウィンドウ ハンドルがあるので、呼び出す前にそれを強制終了してみませんMessageBox()か?

これは私にとってはうまくいきました(テストアプリで):

::SendMessage(hwnd, WM_DESTROY, 0, 0);
MessageBoxW(parent,
    L"This should be MODAL to the main window!\n"
    L"But you should see that in reality the main window\n"
    L"is still enabled!",
    L"mainwin",
    MB_OK | MB_ICONINFORMATION);
于 2015-12-16T17:18:05.383 に答える