虽然没有内置的方法来实现这一点,但您可以尝试使用事件绑定来解决它。在
下面的示例将使用一些绑定来尝试和管理菜单的位置。在
我们需要一个Toplevel()窗口来存放菜单。在
接下来我们需要去掉它的按钮(-,[]和{})。这可以通过overidedirect(True)完成。在
既然我们不能手动移动Toplevel()窗口,我们需要将事件绑定到一个函数,该函数将把顶层放在主窗口的最右边。在
这是绑定:win.bind("", lambda x: win.after(0, move_menu(x)))
函数如下:
^{pr2}$
下一步,我们需要强制菜单或者更确切地说,顶层窗口停留在主窗口的顶部。这可以通过menu_frame.attributes('-topmost', True)完成。然而,当你在manwindow之外点击时,我们会遇到一个问题,那就是顶级菜单窗口停留在tkinter应用程序之外所有程序的顶部。在
为了管理这种行为,我们需要另外两个绑定。一个用于事件'',一个用于事件''。这将允许我们在鼠标进入和离开根窗口时切换overrideredirect()方法。在
以下是绑定:win.bind("", lambda x: win.after(0, manage_top_attr(x, True)))
win.bind("", lambda x: win.after(0, manage_top_attr(x, False)))
函数如下:def manage_top_attr(event, tf):
menu_frame.attributes('-topmost', tf)
所有这些添加到程序中,我们可以在屏幕的右侧有一个菜单。在
代码示例:import tkinter as tk
win = tk.Tk()
win.minsize(200, 200)
win.geometry("250x200")
menu_frame = tk.Toplevel(win)
menu_frame.overrideredirect(True)
menu_frame.attributes('-topmost', True)
tk.Label(win, text="").grid(row=0, column=0)
main_window_frame = tk.Frame(win)
main_window_frame.grid(row=1, column=0, sticky="nsew")
def manage_top_attr(event, tf):
menu_frame.attributes('-topmost', tf)
def move_menu(event):
print (event)
x = (win.winfo_width() - menu_frame.winfo_width())
z = (win.winfo_x(), win.winfo_y())
xx = menu_frame.winfo_width()
menu_frame.geometry('%dx%d+%d+%d' % ((xx), 0, (z[0]+x+8), (z[1]+30)))
win.bind("", lambda x: win.after(0, move_menu(x)))
win.bind("", lambda x: win.after(0, manage_top_attr(x, True)))
win.bind("", lambda x: win.after(0, manage_top_attr(x, False)))
tk.Label(main_window_frame, text="Main window").grid(row=0, column=0)
menuBar = tk.Menu(menu_frame)
menu_frame.config(menu=menuBar)
fileMenu = tk.Menu(menuBar, tearoff=0)
menuBar.add_cascade(label="File", menu=fileMenu)
fileMenu.add_command(label="New")
menuBar.add_cascade(label="Edit", menu=fileMenu)
menuBar.add_cascade(label="Options", menu=fileMenu)
menuBar.add_cascade(label="Help", menu=fileMenu)
win.mainloop()
结果:
调整窗口大小后:
现在它的表现并不完美,可能需要一些工作,但这只是一个开始。在