在这个简单的脚本中,listbox
上Page1
选中的项目会从Page2
保存并打印。 listbox
中的值保存在app_data
中。代码运行,但app_data
中没有记录任何值(打印的值为空)。发生以下异常:
Exception in Tkinter callback Traceback (most recent call last):
File "/usr/lib/python3.4/tkinter/__init__.py", line 1536, in __call__
return self.func(*args) File "filepath/file.py", line 35, in <lambda>
button1 = ttk.Button(self,text="Next Page",command=lambda: controller.show_frame(Page2)
or self.controller.app_data["listbox"]
.set(self.listbox.get(self.listbox.curselection())))
AttributeError: 'Page1' object has no attribute 'listbox'
据我所知,控制器无法识别列表框。我使用super()
查看了可能的解决方案,如Understanding Python super() with init() methods所示,但未取得成功。
问题:
什么会使列表框值正确保存到app_data
?
button1 = ttk.Button(self,text="Next Page"
,command=lambda: controller.show_frame(Page2)
or self.controller.app_data["listbox"]
.set(self.listbox.get(self.listbox.curselection()))
完整代码:
from tkinter import *
from tkinter import ttk
class MyApp(Tk):
def __init__(self):
Tk.__init__(self)
# App data in controller
self.app_data = {"listbox": StringVar()}
container = ttk.Frame(self)
container.pack(side="top", fill="both", expand = True)
self.frames = {}
for F in (Page1, Page2):
frame = F(container, self)
self.frames[F] = frame
frame.grid(row=0, column=0, sticky = NSEW)
self.show_frame(Page1)
def show_frame(self, cont):
frame = self.frames[cont]
frame.tkraise()
class Page1(ttk.Frame):
def __init__(self, parent, controller):
ttk.Frame.__init__(self, parent)
self.controller = controller
listbox = Listbox(self,exportselection=0)
listbox.grid()
for item in [0,1,2,3,4,5]:
listbox.insert(END, item)
button1 = ttk.Button(self,text="Next Page"
,command=lambda: controller.show_frame(Page2)
or self.controller.app_data["listbox"]
.set(self.listbox.get(self.listbox.curselection())))
button1.grid()
class Page2(ttk.Frame):
def __init__(self, parent, controller):
ttk.Frame.__init__(self, parent)
self.controller = controller
ttk.Label(self, text='Next Page').grid(padx=(20,20), pady=(20,20))
button1 = ttk.Button(self, text='Select Page',
command=lambda: controller.show_frame(Page1))
button1.grid()
button2 = ttk.Button(self, text='print value', command=self.print_value)
button2.grid()
def print_value(self):
value = self.controller.app_data["listbox"].get()
print ('The value stored in StartPage some_entry = ' + str(value))
app = MyApp()
app.mainloop()
答案 0 :(得分:1)
您的问题很简单:在button1
定义中,您使用self.listbox
的某些方法,但是当您定义列表框时,您还没有将其定义为Page1
的属性:listbox = Listbox(self,exportselection=0)
(前面没有self.
)。这就是您收到错误AttributeError: 'Page1' object has no attribute 'listbox'
。
有两种方法可以避免这种情况:
如果您需要在__init__
Page1
方法之外的某个位置使用列表框,请将其设为属性:self.listbox = Listbox(self,exportselection=0)
并将所有listbox
替换为self.listbox
{1}}。
如果您在其他任何地方都不需要,只需更改self.listbox
定义中listbox
中的button1
:
button1 = ttk.Button(self,text="Next Page"
,command=lambda: controller.show_frame(Page2)
or self.controller.app_data["listbox"]
.set(listbox.get(listbox.curselection())))