#python #matplotlib
#python #matplotlib
Вопрос:
Я хотел бы иметь возможность перетаскивать файл на мой график Matplotlib и что-то делать с указанным файлом (например, открывать и отображать его!). К сожалению, хотя мой скрипт обрабатывает несколько событий, используя: fig.canvas.mpl_connect('button_release_event', self.btn_release)
и тому подобное, я не могу найти ничего, что вернет путь к файлу, удаленному на нем.
Я не совсем на том уровне, на котором я хочу создавать графические интерфейсы с встроенными графическими графиками, поэтому, если есть решение без этого, я бы предпочел его.
Ответ №1:
Кажется, вы хотите, чтобы a drop_file_event
обрабатывался mpl_connect
; к сожалению, это не так в соответствии с документацией (и в любом случае это случай, немного специфичный для универсальной библиотеки построения графиков).
Однако не так сложно реализовать собственный графический интерфейс, обрабатывающий событие drop. Ниже приведен пример, в основном основанный на embedding_in_wx2 и добавлении a wx.FileDropTarget
.
# Used to guarantee to use at least Wx2.8
import wxversion
wxversion.ensureMinimal('2.8')
import numpy as np
import wx
import matplotlib
matplotlib.use('WXAgg')
from matplotlib.backends.backend_wxagg import FigureCanvasWxAgg as FigureCanvas
from matplotlib.backends.backend_wx import NavigationToolbar2Wx
from matplotlib.figure import Figure
class MyFileDropTarget(wx.FileDropTarget):
def __init__(self, window):
wx.FileDropTarget.__init__(self)
self.window = window
def OnDropFiles(self, x, y, filenames):
fig = self.window.figure
inaxes = fig.get_axes()[0]
h_pix = int(fig.get_figheight() * fig.get_dpi()) # fig height in pixels
message = "%d file(s) dropped at (%d,%d):n" % (len(filenames), x, y)
for file in filenames:
message = file "n"
inaxes.annotate(message, (x, h_pix-y), xycoords='figure pixels')
self.window.draw()
class CanvasPanel(wx.Panel):
def __init__(self, parent):
wx.Panel.__init__(self, parent)
self.figure = Figure()
self.axes = self.figure.add_subplot(111)
self.canvas = FigureCanvas(self, -1, self.figure)
self.sizer = wx.BoxSizer(wx.VERTICAL)
self.sizer.Add(self.canvas, 1, wx.LEFT | wx.TOP | wx.GROW)
self.SetSizer(self.sizer)
self.add_toolbar()
self.Fit()
win_target = self.canvas
dt = MyFileDropTarget(win_target)
win_target.SetDropTarget(dt)
def draw(self):
t = np.linspace(0.0, 2., 100)
s = np.sin(2 * np.pi * t)
self.axes.plot(t, s)
def add_toolbar(self):
self.toolbar = NavigationToolbar2Wx(self.canvas)
self.toolbar.Realize()
if wx.Platform == '__WXMAC__':
# Mac platform (OSX 10.3, MacPython) does not seem to cope with
# having a toolbar in a sizer. This work-around gets the buttons
# back, but at the expense of having the toolbar at the top
self.SetToolBar(self.toolbar)
else:
# On Windows platform, default window size is incorrect, so set
# toolbar width to figure width.
tw, th = self.toolbar.GetSizeTuple()
fw, fh = self.canvas.GetSizeTuple()
# By adding toolbar in sizer, we are able to put it at the bottom
# of the frame - so appearance is closer to GTK version.
# As noted above, doesn't work for Mac.
self.toolbar.SetSize(wx.Size(fw, th))
self.sizer.Add(self.toolbar, 0, wx.LEFT | wx.EXPAND)
# update the axes menu on the toolbar
self.toolbar.update()
if __name__ == "__main__":
app = wx.PySimpleApp()
frame = wx.Frame(None, title='File drop test')
panel = CanvasPanel(frame)
panel.draw()
frame.Show()
app.MainLoop()
Комментарии:
1. Приятно знать, что я ничего не упустил. Мое приключение с графическим интерфейсом начинается раньше, чем я планировал. Спасибо за пример.