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

jgo*_*ers 5 python tk-toolkit tkinter

我正在制作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()
Run Code Online (Sandbox Code Playgroud)

jgo*_*ers 3

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

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

def app_lost_focus(self, event):
    if self.focus_get() != self.menu:
        self.destroy_menu(event)
    
Run Code Online (Sandbox Code Playgroud)