python - 在 Mac OS X 上使用 Tkinter 和 Matplotlib 重复对话窗口

标签 python macos python-2.7 matplotlib tkinter

我是 Tkinter 的新手。我尝试使用下一个代码使用 tkFileDialog.askopenfilename 打开一个文件,然后用 Matplotlib 绘制一些东西:

import matplotlib.pyplot as plt
import Tkinter, tkFileDialog

root = Tkinter.Tk()
root.withdraw()
file_path = tkFileDialog.askopenfilename()

x = range(10)
plt.plot(x)
plt.show()

运行上面的脚本后,我得到一个对话框窗口来打开我的文件。选择文件后,我会重复打开文件的对话窗口和屏幕底部的新窗口。我知道问题出在 plt.show() 上。会发生什么以及如何避免对话窗口重新打开?我应该为我的任务设置一个 Matplotlib 后端吗?

我的版本:

Tcl/Tk 8.5.9

Matplotlib 1.3.1

Tkinter $修订版:81008 $

OS X 10.9.4

我发现了两个相关的 stackoverflow 问题: pyplot-show-reopens-old-tkinter-dialogmatplotlib-figures-not-working-after-tkinter-file-dialog 但没有答案。 root.destroy() 似乎对我不起作用。

最佳答案

当使用 python test.py 运行时,以下似乎对我有用:

import matplotlib.pyplot as plt
import Tkinter, tkFileDialog

root = Tkinter.Tk()
root.withdraw()
file_path = tkFileDialog.askopenfilename()
root.destroy()

print file_path

x = range(10)
plt.plot(x)
plt.show()

我认为它有效,因为文件对话框的 Tk 实例在 matplotlib 启动它自己之前被销毁了。有趣的是,从

ipython --pylab=tk

我预计两次启动事件循环会出现问题。在这种情况下,规范的解决方案是在(再次)启动之前检查 Tk 是否已经在运行。

我在 MacOSX 10.7.5 上,自定义构建的 matplotlib(应该无关紧要)。

我唯一注意到的是,在对此进行试验后,我的 Mac 上的触摸板滑动手势不再有效......调查一下。

编辑

这是 tkFileDialog.askopenfilename() 执行的 Tk 命令的分解:

# breakdown of tkFileDialog.askopenfilename()
import Tkinter as Tk
window = Tk.Tk()
window.withdraw()
w = Tk.Frame(window)

s = w.tk.call('tk_getOpenFile', *w._options({}))
print s

w.destroy()
window.destroy()

当我运行它时(使用 python test.py),我会看到一个文件打开对话框,我可以在其中选择一个文件。 “确定”后,它会打印文件名并退出。这在我的系统上每次都有效。但是,有时我的触摸板上的三指手势在运行该程序时会停止工作!程序退出后他们不会回来!甚至在我杀死程序正在运行的终端之后!!!

我发现将它们带回来的唯一方法是将以下 matplotlib 代码添加到 test.py:

import matplotlib
matplotlib.use('tkagg')
import matplotlib.pyplot as plt
plt.figure()     # simplified from plt.plot(range(10))
plt.show()

然后点击“图1”的标题栏。这会立即带回三指手势。我怀疑这只是 Tkinter 中的一个错误。 (顺便说一句,我使用的是 Tcl/Tk 8.5)

我无法重现文件打开对话框在我的系统上不断重新启动的行为。

如果您在没有任何 matplotlib 命令的情况下启动 test.py,您能否描述一下您的系统会发生什么情况?

或者,由于 Tkinter 很旧而且明显有问题,我可以建议改用 Qt 吗?它不仅看起来更漂亮,而且更敏捷,而且我没有遇到任何错误问题。

编辑2

我已经分解了 matplotlib 在非交互式环境中执行上述命令时所采取的 Tk 操作(即使用 python test.py 而不是来自 iPython)。这些是基本的后端调用:

import matplotlib.backends.backend_tkagg as backend
figManager = backend.new_figure_manager(1)
figManager.show()
backend.show.mainloop()

这些仍然是后端独立的。即,对于 Qt 图,只需使用:

import matplotlib.backends.backend_qt4agg as backend

如果我们进一步将其分解为特定于后端的层,我们有:

import matplotlib.backends.backend_tkagg as backend
import Tkinter as Tk
window = Tk.Tk()
window.withdraw()

# uncomment this to use the same Tk instance for tkFileDialog and matplotlib
#import tkFileDialog
#fname = tkFileDialog.askopenfilename(master=window)
#print fname

# figManager = backend.new_figure_manager(1)
from matplotlib.figure import Figure
figure = Figure()
canvas = backend.FigureCanvasTkAgg(figure, master=window)
figManager = backend.FigureManagerTkAgg(canvas, 1, window)

# figManager.show()
window.deiconify()

# backend.show.mainloop()
Tk.mainloop()

首先,在 tkFileDialog 调用被注释掉的情况下运行,并检查 matplotlib 图形是否出现并正确运行。然后取消对 tkFileDialog 调用的注释,看看您是否最终获得了预期的行为。

如果不是,则必须继续分解 FigureCanvasTkAggFigureManagerTkAgg 以了解发生了什么......

编辑3

OK,既然问题依旧,让我们进一步分解matplotlib的Tk调用。下面的代码完全隔离了我认为必不可少的所有 matplotlib 的 Tk 操作(因此不再需要从 matplotlib 导入任何东西!)。请注意,我没有生成工具栏并分配大量回调和按键事件。如果下面的代码现在可以工作,那么问题就出在这些地方。如果它不起作用,我们可以得出结论,这纯粹是一个 Tk 问题,很可能是一个应该报告的错误。这是代码:

import Tkinter as Tk
window = Tk.Tk()
window.withdraw()

# uncomment this to use the same Tk instance for tkFileDialog and matplotlib
#w = Tk.Frame(window)
#fname = w.tk.call('tk_getOpenFile', *w._options({}))
#print fname
#w.destroy()

# canvas = backend.FigureCanvasTkAgg(figure, master=window)
_tkcanvas = Tk.Canvas(master=window, width=640, height=480, borderwidth=4)
_tkphoto = Tk.PhotoImage(master=_tkcanvas, width=640, height=480)
_tkcanvas.create_image(320, 240, image=_tkphoto)
_tkcanvas.focus_set()

# figManager = backend.FigureManagerTkAgg(canvas, 1, window)
window.wm_title("Figure 1")
window.minsize(480, 360)
_tkcanvas.pack(side=Tk.TOP, fill=Tk.BOTH, expand=1)

# figManager.show()
window.deiconify()

# backend.show.mainloop()
Tk.mainloop()

请尝试注释掉一些行,看看是否可以使其正常工作。如果不是,我会得出结论,这是 Tkinter 中的错误,应该报告。

关于python - 在 Mac OS X 上使用 Tkinter 和 Matplotlib 重复对话窗口,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24660578/

相关文章:

python - 加速 Cython 中的 for 循环

swift - xcrun : error: unable to find utility "xctest", 不是开发人员工具或在 PATH 中

node.js - Node npm install -g Express 生成器

python - 通过 Python 读取 Outlook 事件

python - 有时只读属性

python-2.7 - 按值范围分组数据

python - Chaco Legend 中的自定义标签

python - Pandas 从一列字符串中删除字符

python - 如何使用 Python 正确记录错误/信息?

python - 如何安装 Python Pandas ?