6

我制作了一个可运行的示例来演示错误的行为: http: //pastebin.com/8KpzD4pw


这个问题非常严重。我在保存文件时有一个wx.ProgressDialogup ,然后IOError我想关闭进度对话框并显示一条错误消息。不幸的是,这似乎是不可能的。在消息框关闭之前,进度对话框公然拒绝关闭:

插图

如您所见,消息框出现在进度对话框下方,因此用户必须手动将焦点切换到消息框才能查看其内容。当消息框关闭时,进度对话框也会消失。这是保存功能的代码:

def save(self, path=None):
    # Show a loading dialog while the document is staving.
    progress = shared.show_loading(self, 'Saving document')

    try:
        raise IOError('Error message')

        if not path:
            self.document.save()
        else:
            self.document.save_to_file(path)
    except IOError as e:
        progress.done()

        message = 'Failed to save file:\n\n{}'.format(e.message)
        wx.MessageBox(message, 'Error', wx.OK | wx.ICON_ERROR)

    progress.done()

show_loadingprogress.done函数只是使用wx.ProgressDialog( source )的快捷方式。

为什么在打开消息框之前进度对话框没有消失?我该如何解决?


我也尝试使用wx.CallAfter打开消息框,但无济于事:

# ...
except IOError as e:
    message = 'Failed to save file:\n\n{}'.format(e.message)

    def show_error():
        wx.MessageBox(message, 'Error', wx.OK | wx.ICON_ERROR)

    progress.done()
    wx.CallAfter(show_error)
# ...

我还尝试在关闭进度对话框和使用wx.MicroSleep未成功打开消息框之间睡 100 毫秒。

我也尝试过在销毁进度对话框后立即调用wx.Yield()wx.WakeUpIdle()但都没有任何效果。

4

6 回答 6

5

只是出于好奇...您是否尝试在调用progressdialog.Destroy() 之后立即使用wx.SafeYield() 或wx.Yield() 或wx.YieldIfNeeded()?

您的样本不能按原样运行,所以我只是在黑暗中拍摄。

于 2012-12-06T16:29:19.943 回答
4

我认为 Infinity77 在这里有正确的答案。人们忘记了 GUI 调用不是同步的——它们在返回时还没有完成。那个“完成”调用向窗口发送一条消息,作为响应,窗口可能会再排队几条消息来清理自己。当您启动模型消息框时,它会创建其 OWN 消息循环,同时将原始消息循环保持在暂停状态。因此,在消息框返回并且您的主消息循环再次运行之前,无法处理清理消息。Yield 调用将允许那些排队的消息耗尽。

于 2012-12-06T18:23:15.623 回答
3

我有一个类似的案例,我最终通过调用解决了这个问题:

dlg.Update( dlg.GetRange( ) )

看来,至少当您将进度对话框置于“脉冲”模式时,它不会立即响应 Destroy 调用。在销毁它之前或之后,再多的睡眠或屈服都不会说服我的进度对话框停止显示。但是,通过简单地将值更新为最大值,它似乎会立即自动销毁(或至少隐藏)自身。

于 2019-06-13T21:08:50.977 回答
2

wxPython 演示展示了如何中断 ProgressDialog。它还表明您需要 Destroy() 它而不是 Close() 它,这是摆脱对话框的正常方法。在您的异常处理程序中,您将希望停止 ProgressDialog 正在跟踪的任何内容并 Destroy() 它。然后显示您的 MessageBox。

于 2012-12-06T14:39:53.690 回答
2

我想出了一个解决方法。事实证明,我无法在创建本机 Windows 进度对话框后立即删除它。在我被允许销毁它之前,我必须等待一段时间,可能是让对话框完全初始化。我添加了这段代码:

wx.MilliSleep(50)

进入我的进度对话框快捷功能,它会在打开进度对话框后引入不明显的延迟,并允许我随时销毁进度对话框。

完整的快捷功能:

def show_loading(parent, title, message=None, maximum=100):
    if not message:
        message = title

    # A class for the return value.
    class LoadingObject(object):
        def __init__(self, dialog):
            self.dialog = dialog
            self.is_done = False

        def done(self):
            if not self.is_done:
                self.dialog.Destroy()
                self.is_done = True

        def pulse(self, message):
            self.dialog.Pulse(message)

        def progress(self, current, message=None):
            # Don't allow the progress to reach 100%, since it will freeze the
            # dialog.
            if current >= maximum:
                current = current - 1

            if message is not None:
                self.dialog.Update(current, message)
            else:
                self.dialog.Update(current)

    # Create the progress dialog.
    dlg_style = wx.PD_APP_MODAL | wx.PD_ELAPSED_TIME | wx.PD_REMAINING_TIME
    dlg = wx.ProgressDialog(
        title, message, parent=parent, style=dlg_style, maximum=maximum
    )
    dlg.Pulse()

    # Wait just a little bit to allow the progress dialog to initialize.
    wx.MilliSleep(50)

    # Return an instance of the LoadingDialog with the progress dialog attached.
    return LoadingObject(dlg)

最终保存功能:

def save(self, path=None):
    # Show a loading dialog while the document is staving.
    progress = shared.show_loading(self, 'Saving document')

    try:
        if not path:
            self.document.save()
        else:
            self.document.save_to_file(path)
    except IOError as e:
        message = 'Failed to save file:\n\n{}'.format(e.message)
        wx.MessageBox(message, 'Error', wx.OK | wx.ICON_ERROR)
    finally:
        progress.done()
于 2012-12-07T07:39:34.820 回答
-2

我对这个问题采取了稍微不同的方法。以线性顺序进行了多个函数调用,并希望在函数调用完成时显示整体进度。我只是将进度条包装在一个函数中,获取要调用的函数及其参数。在异常情况下,我会破坏进度条,并引发异常。下面的例子:

def _progress_wrap(self, func, *args, **kwargs):
    self.count += 1
    self.progress.Update(self.count)
    res = None
    try:
        res = func(*args, **kwargs)
    except Exception:
        self.progress.Destroy()
        raise
    return(res)
于 2014-01-29T09:00:44.743 回答