简体   繁体   中英

Python Tkinter: Bind function with labels in for loop

I'm creating labels dynamically in a for loop using tkinter . I don't know how many labels will be created, but on clicking of each of the labels, a particular function must be called with a particular parameter.

To do this, I'm using this code:

for link in list_of_links:
    link_label = Label(self.video_window, text="Frame "+str(video_number), fg="blue", cursor="hand2")
    link_label.pack()
    link_label.place(x=xcod2, y=ycod2)
    link_label.bind("<1>", lambda x: self.goto_video_link(link))

Currently, I'm creating 10 labels. The problem is that on clicking any of the ten labels, the goto_video_link function seems to only use the 10th link.

If I click on the 5th label, I want it to use the 5th link.

How do I go about this?

Lambda expressions are lazily evaluated, which means that self.go_to_link(link) is only evaluated when it is executed. In this moment link contains the value of the last link, so every button will go to the last link.

You need to force the evaluation of link during the for loop. This can be done with a lambda function that returns another lambda function with the value you want. I know it seems confusing, but the code below may make it clearer.

eval_link = lambda x: (lambda p: self.go_to_link(x))
for link in list_of_links:
    link_label = Label(self.video_window, text="Frame "+str(video_number), fg="blue", cursor="hand2")
    link_label.pack()
    link_label.place(x=xcod2, y=ycod2)
    link_label.bind("<1>", eval_link(link))

In this case, to be able to build the inner lambda it is necessary to evaluate link . Since it gets passed as a parameter, the inner most lambda is bound to the local copy x instead of link and since x is a local variable, it is always remade when the function is called.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM