2

使用 WinJS,我试图向刚刚点击“继续”按钮的用户显示一条消息,表明他们没有回答所提出的问题。这将显示一个带有两个按钮的对话框(Windows.UI.Popups.MessageDialog),允许用户选择现在回答(因此留在未回答的问题上)或稍后回答(因此直接进入下一个问题) . 在 Javascript 中,我通常可以使用确认对话框,它会暂停执行,直到用户做出选择,然后我可以使用返回的值来决定是否显示下一个问题。在 WinJS 中,代码执行不会停止。这是我显示对话框的功能:

function doSillyMessage(message, title) {
    var messagedialogpopup = new Windows.UI.Popups.MessageDialog(message, title);
    // Add commands and set their CommandIds
    messagedialogpopup.commands.append(new Windows.UI.Popups.UICommand("Answer now", null, 1));
    messagedialogpopup.commands.append(new Windows.UI.Popups.UICommand("Answer later", null, 2));

    // Set the command that will be invoked by default
    messagedialogpopup.defaultCommandIndex = 0;

    // Show the message dialog
    messagedialogpopup.showAsync().done(function (command) {
        if (command) {
            if (command.id == 1) {
                console.log('proceed in popup is false');
                proceed = false;
            }
            else if (command.id == 2) {
                console.log('proceed in popup is true');
                proceed = true;
            }

        }
    });
}

问题在于,在调用 doSillyMessage 时,代码不会等待,因此会继续处理“proceed”变量的当前值。例如,如果在调用 doSillyMessage 时 'proceed' 的值为 false,则在调用 doSillyMessage 后它也是 false 并且仅在稍后更改为 true(如果合适)时,然后我在控制台中看到 'proceed in弹出是真的'。我怎样才能让这些东西以正确的顺序执行?

4

2 回答 2

5

对话框是异步的;您需要等待承诺才能延迟到用户处理输入。

最简单的做法是从您对 showAsync 的调用中返回 Promise。改变这个:

messagedialogpopup.showAsync().done(function (command) {

return messagedialogpopup.showAsync().then(function (command) {
   if (command) {
        if (command.id == 1) {
            console.log('proceed in popup is false');
            return false;
        }
        else if (command.id == 2) {
            console.log('proceed in popup is true');
            return true;
        }
    }
});

并将对 doSillyMessage 的调用更改为:

doSillyMessage().then(function (proceed) {
    // proceed will be true or false
});

您的回调逻辑中有一些漏洞;如果没有设置命令,或者不是 1 或 2,你会返回什么?

于 2013-04-08T00:13:12.117 回答
2

我最终编写了一个对话框助手集来管理我的确认、警报等。这可能对你有用:

帮手

function confirmAsync(title, msg) {

    // define confirmation buttons
    var buttons = [
        new Windows.UI.Popups.UICommand("Yes"),
        new Windows.UI.Popups.UICommand("No")];

    // make sure to return the showAsync() promise so we can chain from the call to confirmAsync
    return dialogInterceptor(title, msg, buttons).showAsync();

}


function dialogInterceptor(title, msg, buttons) {

    // create dialog
    var dialog = new Windows.UI.Popups.MessageDialog(msg, title);

    // optionally add buttons
    if (buttons) {

        buttons.forEach(function(button) {

            dialog.commands.append(button);

        });

    }

    // return dialog
    return dialog;
}

用法:

            confirmAsync("Confirmation title text", "Are you sure you want to do something?")
                .done(function (button) {

                    if (button.label === "Yes") {

                        functionToCallWhenYesIsClicked();
                    }

            });

不过,一般来说,从你的对话中返回承诺,以便你以后可以做一些事情。尝试从第一个对话框启动第二个对话框时要小心,否则您会遇到讨厌的“拒绝访问”消息。

于 2013-06-13T07:55:06.260 回答