在事件期间,我似乎无法设置根为ttk.Label
的ttk.Entry
的文本:
import tkinter
from tkinter import ttk
def modify_label_text(event):
entry = event.widget
newvalue = entry.get()
label = entry.master
label.configure(text=newvalue)
entry.unbind("<Return>")
entry.pack_forget()
label.focus()
def create_entry(event):
label = event.widget
oldvalue = label.cget("text")
newvalue = tkinter.StringVar()
entry = ttk.Entry(label, textvariable=newvalue)
'''
entry = tkinter.Entry(label, textvariable=newvalue)
'''
entry.pack()
entry.delete(0, "end")
entry.insert(0, oldvalue)
entry.bind("<Return>", modify_label_text)
entry.focus()
root = tkinter.Tk()
clickme = ttk.Label(root, width=16)
clickme.pack()
clickme.bind("<Button-1>", create_entry)
clickme.focus()
root.mainloop()
单击空标签并输入新值时,该值将反映在标签中。然而,如果我再次点击标签";编辑";值,则entry字段再次为空。
此外,如果我使用tkinter.Entry
而不是ttk.Entry
,它似乎可以工作。
为什么只有在使用tkinter.Entry
时才设置条目的文本?如何修复此问题以使用ttk.Entry
?
这可能不是一个答案,但对于注释来说有点太大了。也可能有人在不同的环境中详细描述这种行为。我用运行了那个代码
Mashine:
Windows 10 Home; x64-base,Intel(R) Core(TM) i3-2120 @ 3.30GHz, 3300 MHz, 2Cores
带有
Python 3.7.2 and tkinter 8.6
经过一些研究,我找不出这种行为的原因。无论是在文档中,还是在有时棘手的style_map或默认绑定中。
我可以假设您使用函数/循环中的条目创建标签吗?您可能想要重用该条目,因为出于某种原因,如果您通过winfo_children()
寻址该条目,则完全相同的代码也可以工作。即使我叫.update_idletasks()
,或者只是为了体验邪恶的.update()
,它也不会无缘无故地起作用。
import tkinter
from tkinter import ttk
def modify_label_text(event):
entry = event.widget
newvalue = entry.get()
label = entry.master
label.configure(text=newvalue)
entry.unbind("<Return>")
entry.pack_forget()
label.focus()
def create_entry(event):
label = event.widget
oldvalue = label.cget("text")
newvalue = tkinter.StringVar()
if len(label.winfo_children()) == 0:
print('new')
entry = ttk.Entry(label, textvariable=newvalue)
'''
entry = tkinter.Entry(label, textvariable=newvalue)
'''
e = label.winfo_children()[0]
e.pack()
e.delete(0, "end")
e.insert(0, oldvalue+' works')
e.bind("<Return>", modify_label_text)
e.focus()
root = tkinter.Tk()
clickme = ttk.Label(root, width=16)
clickme.pack()
clickme.bind("<Button-1>", create_entry)
clickme.focus()
root.mainloop()
另一种解决方案可能是:
def create_entry(event):
label = event.widget
oldvalue = label.cget("text")
newvalue = tkinter.StringVar()
entry = ttk.Entry(label, textvariable=newvalue)
'''
entry = tkinter.Entry(label, textvariable=newvalue)
'''
root.after(100,after_time,entry,oldvalue)
def after_time(entry,oldvalue):
entry.pack()
entry.delete(0, "end")
entry.insert(0, oldvalue)
entry.bind("<Return>", modify_label_text)
entry.focus()
我建议您在函数外创建一个Entry实例,并在函数中简单地创建pack()
和pack_forget()
。pack_forget()
不会删除您的小部件,只会在布局中隐藏它。要删除小部件,您可能必须使用widget.destroy()
。
import tkinter
from tkinter import ttk
def modify_label_text(event):
newvalue = entry.get()
clickme.configure(text=newvalue)
entry.pack_forget()
clickme.focus()
def create_entry(event):
label = event.widget
entry.pack()
entry.focus()
root = tkinter.Tk()
newvalue = tkinter.StringVar()
clickme = ttk.Label(root, width=16)
clickme.pack()
clickme.bind("<Button-1>", create_entry)
clickme.focus()
entry = ttk.Entry(clickme, textvariable=newvalue)
entry.bind("<Return>", modify_label_text)
root.mainloop()
编辑(将我的评论作为答案(
我怀疑它与正在设置的文本变量有关。简单地删除它或将newvalue更改为全局范围似乎是可行的。此外,由于您似乎没有使用.set
或.get
方法,因此如果需要,可以简单地将其删除。