tkinter 中鼠标滚轮 + 滚动条的问题
trouble with mousewheel + scrollbars in tkinter
我正在尝试使用 tkinter 显示图像,但在添加滚动条时遇到了问题。如果图像超过窗口大小,我希望能够水平和垂直滚动鼠标滚轮。问题是当我使用 canvas.
时鼠标滚轮不起作用
在此代码中,我可以使用鼠标滚轮滚动 Y 并使用 shift+鼠标滚轮滚动 X(这正是我想要的):
import tkinter as tk
master = tk.Tk()
scrollbarX = tk.Scrollbar(master)
scrollbarX.pack(side=tk.BOTTOM, fill=tk.X)
scrollbarY = tk.Scrollbar(master)
scrollbarY.pack(side=tk.RIGHT, fill=tk.Y)
listbox = tk.Listbox(master, yscrollcommand=scrollbarY.set, xscrollcommand=scrollbarX.set)
for i in range(1000):
listbox.insert(tk.END, str(i)+"-----------------------------------------------------------------------")
listbox.pack(side=tk.LEFT, fill=tk.BOTH, expand=tk.YES)
scrollbarX.config(command=listbox.xview, orient=tk.HORIZONTAL)
scrollbarY.config(command=listbox.yview)
master.mainloop()
但是,如果我将列表框替换为 canvas,鼠标滚轮将不再执行任何操作。手动滚动仍然有效:
import tkinter as tk
from PIL import Image, ImageTk
master = tk.Tk()
scrollbarX = tk.Scrollbar(master)
scrollbarX.pack(side=tk.BOTTOM, fill=tk.X)
scrollbarY = tk.Scrollbar(master)
scrollbarY.pack(side=tk.RIGHT, fill=tk.Y)
canvas = tk.Canvas(master, bd=0, xscrollcommand=scrollbarX.set, yscrollcommand=scrollbarY.set)
imagefile = "image.png"
img = ImageTk.PhotoImage(Image.open(imagefile))
canvas.create_image(0,0,image=img, anchor="nw")
canvas.pack(fill=tk.BOTH, expand=tk.YES)
scrollbarX.config(command=canvas.xview, orient=tk.HORIZONTAL)
scrollbarY.config(command=canvas.yview)
canvas.config(scrollregion=canvas.bbox(tk.ALL))
master.mainloop()
我还是编程新手,非常感谢任何帮助!
您需要处理鼠标事件并进行滚动。
您可以通过这样的回调来做到这一点:
def on_mousewheel(event):
shift = (event.state & 0x1) != 0
scroll = -1 if event.delta > 0 else 1
if shift:
canvas.xview_scroll(scroll, "units")
else:
canvas.yview_scroll(scroll, "units")
然后像这样将它绑定到 canvas
:
canvas.bind_all("<MouseWheel>", on_mousewheel)
我正在尝试使用 tkinter 显示图像,但在添加滚动条时遇到了问题。如果图像超过窗口大小,我希望能够水平和垂直滚动鼠标滚轮。问题是当我使用 canvas.
时鼠标滚轮不起作用在此代码中,我可以使用鼠标滚轮滚动 Y 并使用 shift+鼠标滚轮滚动 X(这正是我想要的):
import tkinter as tk
master = tk.Tk()
scrollbarX = tk.Scrollbar(master)
scrollbarX.pack(side=tk.BOTTOM, fill=tk.X)
scrollbarY = tk.Scrollbar(master)
scrollbarY.pack(side=tk.RIGHT, fill=tk.Y)
listbox = tk.Listbox(master, yscrollcommand=scrollbarY.set, xscrollcommand=scrollbarX.set)
for i in range(1000):
listbox.insert(tk.END, str(i)+"-----------------------------------------------------------------------")
listbox.pack(side=tk.LEFT, fill=tk.BOTH, expand=tk.YES)
scrollbarX.config(command=listbox.xview, orient=tk.HORIZONTAL)
scrollbarY.config(command=listbox.yview)
master.mainloop()
但是,如果我将列表框替换为 canvas,鼠标滚轮将不再执行任何操作。手动滚动仍然有效:
import tkinter as tk
from PIL import Image, ImageTk
master = tk.Tk()
scrollbarX = tk.Scrollbar(master)
scrollbarX.pack(side=tk.BOTTOM, fill=tk.X)
scrollbarY = tk.Scrollbar(master)
scrollbarY.pack(side=tk.RIGHT, fill=tk.Y)
canvas = tk.Canvas(master, bd=0, xscrollcommand=scrollbarX.set, yscrollcommand=scrollbarY.set)
imagefile = "image.png"
img = ImageTk.PhotoImage(Image.open(imagefile))
canvas.create_image(0,0,image=img, anchor="nw")
canvas.pack(fill=tk.BOTH, expand=tk.YES)
scrollbarX.config(command=canvas.xview, orient=tk.HORIZONTAL)
scrollbarY.config(command=canvas.yview)
canvas.config(scrollregion=canvas.bbox(tk.ALL))
master.mainloop()
我还是编程新手,非常感谢任何帮助!
您需要处理鼠标事件并进行滚动。
您可以通过这样的回调来做到这一点:
def on_mousewheel(event):
shift = (event.state & 0x1) != 0
scroll = -1 if event.delta > 0 else 1
if shift:
canvas.xview_scroll(scroll, "units")
else:
canvas.yview_scroll(scroll, "units")
然后像这样将它绑定到 canvas
:
canvas.bind_all("<MouseWheel>", on_mousewheel)