Tk / Tkinter:检测应用程序丢失焦点

时间:2013-08-06 19:42:15

标签: python tkinter tk

我正在制作Tkinter应用程序。在应用程序中,我想弹出一个上下文菜单,我使用Tk.Menu.post()。

我不知道当应用程序失去焦点时如何使此菜单消失。我需要这样做,因为菜单保持在顶部,即使切换到另一个窗口,留下菜单'伪影。

我在菜单上放了一个<FocusOut>事件,如果菜单具有焦点并且用户将焦点移动到另一个应用程序,则会触发该事件。这很有效。

如果主应用程序窗口具有焦点,我该怎么办?我可以在应用程序窗口中放置一个<FocusOut>事件,关闭菜单;然而,当我将焦点放在关闭菜单的菜单上时,最终会被调用。菜单是使用parent作为主应用程序创建的,因此我不确定当菜单获得焦点时,主应用程序上的<FocusOut>甚至会被触发。

如何区分主要应用程序窗口失去焦点到不同的应用程序而失去焦点到我的菜单?

我不想使用tk_popup()因为我希望用户继续向主窗口提供输入。 (使用菜单是可选的)。

感谢@Brad Lanam,我提出了一个解决方案,其中包括:

from Tkinter import *

class App(Tk):
    def __init__(self, *args, **kwargs):
        Tk.__init__(self, *args, **kwargs)

        self.entry = Entry(self)
        self.entry.grid(padx=30, pady=30)
        self.entry.focus_set()
        self.entry.bind("<Tab>", self.put_menu)
        self.entry.bind("<Down>", self.set_focus_on_menu)

        self.menu = Menu(self, tearoff=False)
        self.menu.add_command(label="test")
        self.menu.bind("<FocusOut>", self.destroy_menu)


        self.bind("<FocusIn>", self.app_got_focus)
        self.bind("<FocusOut>", self.app_lost_focus)
        self.bind("<3>", self.put_menu)


    def app_got_focus(self, event):
        self.config(background="red")

    def app_lost_focus(self, event):
        self.config(background="grey")

        ######## SOLUTION BEGIN #########
        if self.focus_get() != self.menu:
            self.destroy_menu(event)
        ######## SOLUTION END ###########

    def set_focus_on_menu(self, event):
        self.menu.focus_set()

    def menu_got_focus(self, event):
        self.menu.activate(0)

    def put_menu(self, event):
        self.menu.post(self.winfo_x() + self.entry.winfo_x(), self.winfo_y() + self.entry.winfo_y()+20)

    def destroy_menu(self, event):
        self.menu.destroy()

app = App()

app.mainloop()

1 个答案:

答案 0 :(得分:2)

self.focus_get()将返回具有焦点的对象,该对象可用于区分菜单接收焦点和其他应用程序。

例如,要在焦点移动到另一个应用程序时删除菜单:

def app_lost_focus(self, event):
    if self.focus_get() != self.menu:
        self.destroy_menu(event)